diff --git a/.github/localflows/cicd-local.yml b/.github/localflows/cicd-local.yml
index 9e419470684..beef2f90674 100644
--- a/.github/localflows/cicd-local.yml
+++ b/.github/localflows/cicd-local.yml
@@ -10,9 +10,6 @@ jobs:
- uses: actions/checkout@v3
- name: make test
run: |
- mkdir -p ./bin
- cp -r /go/bin/controller-gen ./bin/controller-gen
- cp -r /go/bin/setup-envtest ./bin/setup-envtest
make mod-vendor lint test
diff --git a/.github/workflows/cicd-pull-request.yml b/.github/workflows/cicd-pull-request.yml
index cf31693b5a6..66bd6e4e9c3 100644
--- a/.github/workflows/cicd-pull-request.yml
+++ b/.github/workflows/cicd-pull-request.yml
@@ -94,9 +94,6 @@ jobs:
- uses: actions/checkout@v3
- name: make mod-vendor
run: |
- mkdir -p ./bin
- cp -r /go/bin/controller-gen ./bin/controller-gen
- cp -r /go/bin/setup-envtest ./bin/setup-envtest
make mod-vendor
- name: make lint
@@ -149,7 +146,7 @@ jobs:
with:
MAKE_OPS_PRE: "generate"
IMG: "apecloud/kubeblocks"
- GO_VERSION: "1.20.5"
+ GO_VERSION: "1.21"
BUILDX_PLATFORMS: "linux/amd64"
DOCKERFILE_PATH: "./docker/Dockerfile"
secrets: inherit
@@ -159,9 +156,9 @@ jobs:
if: contains(needs.trigger-mode.outputs.trigger-mode, '[docker]')
uses: apecloud/apecloud-cd/.github/workflows/release-image-check.yml@v0.1.24
with:
- MAKE_OPS_PRE: "generate test-go-generate"
+ MAKE_OPS_PRE: "module generate test-go-generate"
IMG: "apecloud/kubeblocks-tools"
- GO_VERSION: "1.20.5"
+ GO_VERSION: "1.21"
BUILDX_PLATFORMS: "linux/amd64"
DOCKERFILE_PATH: "./docker/Dockerfile-tools"
secrets: inherit
diff --git a/.github/workflows/cicd-push.yml b/.github/workflows/cicd-push.yml
index d90c6002772..d682d5beb6d 100644
--- a/.github/workflows/cicd-push.yml
+++ b/.github/workflows/cicd-push.yml
@@ -11,7 +11,7 @@ on:
env:
GITHUB_TOKEN: ${{ secrets.PERSONAL_ACCESS_TOKEN }}
BASE_BRANCH: origin/main
- GO_VERSION: "1.20.5"
+ GO_VERSION: "1.21"
jobs:
trigger-mode:
@@ -133,10 +133,6 @@ jobs:
- uses: actions/checkout@v3
- name: make manifests check
run: |
- mkdir -p ./bin
- cp -r /go/bin/controller-gen ./bin/controller-gen
- cp -r /go/bin/setup-envtest ./bin/setup-envtest
-
make manifests
FILE_CHANGES=`git diff --name-only ${{ github.sha }}`
if [[ ! -z "$FILE_CHANGES" ]]; then
@@ -199,7 +195,7 @@ jobs:
with:
MAKE_OPS_PRE: "generate"
IMG: "apecloud/kubeblocks"
- GO_VERSION: "1.20.5"
+ GO_VERSION: "1.21"
BUILDX_PLATFORMS: "linux/amd64"
DOCKERFILE_PATH: "./docker/Dockerfile"
secrets: inherit
@@ -209,9 +205,9 @@ jobs:
if: ${{ contains(needs.trigger-mode.outputs.trigger-mode, '[docker]') }}
uses: apecloud/apecloud-cd/.github/workflows/release-image-check.yml@v0.1.24
with:
- MAKE_OPS_PRE: "generate test-go-generate"
+ MAKE_OPS_PRE: "module generate test-go-generate"
IMG: "apecloud/kubeblocks-tools"
- GO_VERSION: "1.20.5"
+ GO_VERSION: "1.21"
BUILDX_PLATFORMS: "linux/amd64"
DOCKERFILE_PATH: "./docker/Dockerfile-tools"
secrets: inherit
diff --git a/.github/workflows/codeql.yml b/.github/workflows/codeql.yml
index 31a0c296350..1b38f2a4bfb 100644
--- a/.github/workflows/codeql.yml
+++ b/.github/workflows/codeql.yml
@@ -8,7 +8,7 @@ on:
env:
BASE_BRANCH: origin/main
- GO_VERSION: "1.20.5"
+ GO_VERSION: "1.21"
jobs:
trigger-mode:
diff --git a/.github/workflows/e2e-kbcli.yml b/.github/workflows/e2e-kbcli.yml
index 25ec535d1cd..0528bc9728d 100644
--- a/.github/workflows/e2e-kbcli.yml
+++ b/.github/workflows/e2e-kbcli.yml
@@ -12,7 +12,7 @@ on:
required: false
default: ''
TEST_TYPE:
- description: 'test type (e.g. mysql|postgres|redis|mongo|kafka|pulsar|weaviate|qdrant|smarte|scale|greptimedb|nebula|risingwave|starrocks|oceanbase|foxlake)'
+ description: 'test type (e.g. mysql|postgres|redis|mongo|kafka|pulsar|weaviate|qdrant|smarte|scale|greptimedb|nebula|risingwave|starrocks|oceanbase|foxlake|oracle-mysql|asmysql|openldap)'
required: false
default: ''
CLOUD_PROVIDER:
@@ -87,7 +87,7 @@ jobs:
e2e:
name: ${{ inputs.CLOUD_PROVIDER }}
needs: check
- uses: apecloud/apecloud-cd/.github/workflows/kbcli-test-k8s.yml@v0.1.25
+ uses: apecloud/apecloud-cd/.github/workflows/kbcli-test-k8s.yml@main
with:
CLOUD_PROVIDER: "${{ inputs.CLOUD_PROVIDER }}"
KB_VERSION: "${{ needs.check.outputs.release-version }}"
@@ -96,7 +96,7 @@ jobs:
INSTANCE_TYPE: "${{ inputs.INSTANCE_TYPE }}"
REGION: "${{ needs.check.outputs.cluster-region }}"
BRANCH_NAME: "${{ inputs.BRANCH_NAME }}"
- APECD_REF: "v0.1.25"
+ APECD_REF: "main"
ARGS: "${{ inputs.ARGS }}"
TEST_TYPE: "${{ inputs.TEST_TYPE }}"
secrets: inherit
diff --git a/.github/workflows/e2e-performance.yml b/.github/workflows/e2e-performance.yml
index 87ce6c20da4..720e1f5ce5c 100644
--- a/.github/workflows/e2e-performance.yml
+++ b/.github/workflows/e2e-performance.yml
@@ -68,7 +68,7 @@ jobs:
performance:
name: ${{ inputs.NODE_TYPE }} ${{ inputs.PERFORMANCE_TYPE }}
needs: check
- uses: apecloud/apecloud-cd/.github/workflows/performance-test-k8s.yaml@v0.1.23
+ uses: apecloud/apecloud-cd/.github/workflows/performance-test-k8s.yml@v0.1.30
with:
CLOUD_PROVIDER: "eks"
CLUSTER_VERSION: "${{ inputs.K8S_VERSION }}"
@@ -80,5 +80,5 @@ jobs:
BENCH_TABLES: "${{ inputs.BENCH_TABLES }}"
CLUSTER_STORAGE: "${{ inputs.CLUSTER_STORAGE }}"
REGION: "${{ vars.REGION_AWK_EKS }}"
- APECD_REF: "v0.1.23"
+ APECD_REF: "v0.1.30"
secrets: inherit
diff --git a/.github/workflows/milestone-set.yaml b/.github/workflows/milestone-set.yaml
new file mode 100644
index 00000000000..12dde122577
--- /dev/null
+++ b/.github/workflows/milestone-set.yaml
@@ -0,0 +1,35 @@
+name: Set Milestone
+
+on:
+ workflow_dispatch:
+ issues:
+ types:
+ - opened
+ - closed
+ pull_request_target:
+ types:
+ - opened
+ - closed
+
+
+jobs:
+ issue-milestone:
+ if: ${{ github.event_name == 'issues' }}
+ uses: apecloud/apecloud-cd/.github/workflows/issue-milestone.yml@v0.1.31
+ with:
+ APECD_REF: "v0.1.31"
+ secrets: inherit
+
+ pr-milestone:
+ if: ${{ github.event_name == 'pull_request_target' }}
+ uses: apecloud/apecloud-cd/.github/workflows/pull-request-milestone.yml@v0.1.31
+ with:
+ APECD_REF: "v0.1.31"
+ secrets: inherit
+
+ move_milestone:
+ if: ${{ github.event_name == 'workflow_dispatch' }}
+ uses: apecloud/apecloud-cd/.github/workflows/milestone-move.yml@v0.1.31
+ with:
+ APECD_REF: "v0.1.31"
+ secrets: inherit
diff --git a/.github/workflows/milestoneclose.yml b/.github/workflows/milestoneclose.yml
index b0d549aaa9f..b4d1d5750f4 100644
--- a/.github/workflows/milestoneclose.yml
+++ b/.github/workflows/milestoneclose.yml
@@ -12,7 +12,13 @@ env:
PROJECT_NUMBER: 2
jobs:
- move_issues:
+ move_milestone:
+ uses: apecloud/apecloud-cd/.github/workflows/milestone-move.yml@v0.1.31
+ with:
+ APECD_REF: "v0.1.31"
+ secrets: inherit
+
+ move_issues:
runs-on: ubuntu-latest
steps:
- name: Checkout repo
diff --git a/.github/workflows/pull-request-label-size.yaml b/.github/workflows/pull-request-label-size.yaml
index 243487f0f21..775727450ea 100644
--- a/.github/workflows/pull-request-label-size.yaml
+++ b/.github/workflows/pull-request-label-size.yaml
@@ -4,16 +4,10 @@ on:
pull_request_target:
types: [ edited, opened, synchronize ]
-env:
- GITHUB_TOKEN: ${{ github.token }}
jobs:
size-label:
- runs-on: ubuntu-latest
- steps:
- - uses: actions/checkout@v3
- - name: set size label
- run: |
- bash .github/utils/utils.sh --type 15 \
- --github-token "${{ env.GITHUB_TOKEN }}" \
- --pr-number ${{ github.event.pull_request.number }}
+ uses: apecloud/apecloud-cd/.github/workflows/pull-request-label-size.yml@v0.1.30
+ with:
+ APECD_REF: "v0.1.30"
+ secrets: inherit
diff --git a/.github/workflows/release-delete.yml b/.github/workflows/release-delete.yml
index 7593b5aafed..2cc5cf98934 100644
--- a/.github/workflows/release-delete.yml
+++ b/.github/workflows/release-delete.yml
@@ -15,15 +15,15 @@ run-name: Delete Release:${{ inputs.release-version }}
jobs:
delete-release:
if: github.event_name != 'schedule'
- uses: apecloud/apecloud-cd/.github/workflows/release-delete.yml@v0.1.25
+ uses: apecloud/apecloud-cd/.github/workflows/release-delete.yml@v0.1.29
with:
VERSION: "${{ inputs.release-version }}"
- APECD_REF: "v0.1.25"
+ APECD_REF: "v0.1.29"
secrets: inherit
delete-release-schedule:
if: github.event_name == 'schedule'
- uses: apecloud/apecloud-cd/.github/workflows/release-delete-schedule.yml@v0.1.25
+ uses: apecloud/apecloud-cd/.github/workflows/release-delete-schedule.yml@v0.1.29
with:
- APECD_REF: "v0.1.25"
+ APECD_REF: "v0.1.29"
secrets: inherit
diff --git a/.github/workflows/release-helm-chart.yml b/.github/workflows/release-helm-chart.yml
index 6b8e73f8cc9..e46fb800de0 100644
--- a/.github/workflows/release-helm-chart.yml
+++ b/.github/workflows/release-helm-chart.yml
@@ -37,14 +37,14 @@ jobs:
release-chart:
needs: chart-version
- uses: apecloud/apecloud-cd/.github/workflows/release-charts.yml@v0.1.25
+ uses: apecloud/apecloud-cd/.github/workflows/release-charts.yml@v0.1.28
with:
MAKE_OPS: "bump-chart-ver"
VERSION: "${{ needs.chart-version.outputs.chart-version }}"
CHART_NAME: "kubeblocks"
CHART_DIR: "deploy/helm"
DEP_CHART_DIR: "deploy/helm/depend-charts"
- APECD_REF: "v0.1.25"
+ APECD_REF: "v0.1.28"
secrets: inherit
release-charts-image:
@@ -54,7 +54,7 @@ jobs:
MAKE_OPS_PRE: "helm-package VERSION=${{ needs.chart-version.outputs.chart-version-bump }}"
IMG: "apecloud/kubeblocks-charts"
VERSION: "${{ needs.chart-version.outputs.chart-version }}"
- GO_VERSION: "1.20.5"
+ GO_VERSION: "1.21"
APECD_REF: "v0.1.24"
DOCKERFILE_PATH: "./docker/Dockerfile-charts"
secrets: inherit
diff --git a/.github/workflows/release-image.yml b/.github/workflows/release-image.yml
index 4e7bf9d298a..00f1b4c62b2 100644
--- a/.github/workflows/release-image.yml
+++ b/.github/workflows/release-image.yml
@@ -45,7 +45,7 @@ jobs:
MAKE_OPS_PRE: "generate"
IMG: "apecloud/kubeblocks"
VERSION: "${{ needs.image-tag.outputs.tag-name }}"
- GO_VERSION: "1.20.5"
+ GO_VERSION: "1.21"
APECD_REF: "v0.1.24"
DOCKERFILE_PATH: "./docker/Dockerfile"
secrets: inherit
@@ -54,10 +54,10 @@ jobs:
needs: image-tag
uses: apecloud/apecloud-cd/.github/workflows/release-image-cache.yml@v0.1.24
with:
- MAKE_OPS_PRE: "generate test-go-generate"
+ MAKE_OPS_PRE: "module generate test-go-generate"
IMG: "apecloud/kubeblocks-tools"
VERSION: "${{ needs.image-tag.outputs.tag-name }}"
- GO_VERSION: "1.20.5"
+ GO_VERSION: "1.21"
APECD_REF: "v0.1.24"
DOCKERFILE_PATH: "./docker/Dockerfile-tools"
secrets: inherit
diff --git a/.github/workflows/release-publish.yml b/.github/workflows/release-publish.yml
index 6673ad9997f..4fe291849de 100644
--- a/.github/workflows/release-publish.yml
+++ b/.github/workflows/release-publish.yml
@@ -9,7 +9,7 @@ env:
GH_TOKEN: ${{ github.token }}
GITHUB_TOKEN: ${{ secrets.PERSONAL_ACCESS_TOKEN }}
TAG_NAME: ${{ github.ref_name }}
- GO_VERSION: "1.20.5"
+ GO_VERSION: "1.21"
CLI_NAME: 'kbcli'
CLI_REPO: 'apecloud/kbcli'
GITLAB_KBCLI_PROJECT_ID: 85948
diff --git a/.goreleaser.yaml b/.goreleaser.yaml
index 2fa8e063069..696d3c0bb3c 100644
--- a/.goreleaser.yaml
+++ b/.goreleaser.yaml
@@ -28,7 +28,7 @@ builds:
- amd64
- arm64
env:
- - ASSUME_NO_MOVING_GC_UNSAFE_RISK_IT_WITH=go1.20
+ - ASSUME_NO_MOVING_GC_UNSAFE_RISK_IT_WITH=go1.21
- CGO_ENABLED=0
tags:
- containers_image_openpgp
diff --git a/Makefile b/Makefile
index a67ec7d53b6..a85828a59a4 100644
--- a/Makefile
+++ b/Makefile
@@ -477,8 +477,8 @@ CONTROLLER_GEN ?= $(LOCALBIN)/controller-gen
ENVTEST ?= $(LOCALBIN)/setup-envtest
## Tool Versions
-KUSTOMIZE_VERSION ?= v4.5.7
-CONTROLLER_TOOLS_VERSION ?= v0.9.0
+KUSTOMIZE_VERSION ?= v5.1.1
+CONTROLLER_TOOLS_VERSION ?= v0.12.1
CUE_VERSION ?= v0.4.3
KUSTOMIZE_INSTALL_SCRIPT ?= "$(GITHUB_PROXY)https://raw.githubusercontent.com/kubernetes-sigs/kustomize/master/hack/install_kustomize.sh"
@@ -646,8 +646,24 @@ else ifeq ($(TEST_TYPE), kafka)
$(HELM) template kafka-cluster deploy/kafka-cluster > test/e2e/testdata/smoketest/kafka/00_kafkacluster.yaml
else ifeq ($(TEST_TYPE), foxlake)
$(HELM) dependency build deploy/foxlake-cluster --skip-refresh
- $(HELM) upgrade --install etcd deploy/foxlake
+ $(HELM) upgrade --install foxlake deploy/foxlake
$(HELM) template foxlake-cluster deploy/foxlake-cluster > test/e2e/testdata/smoketest/foxlake/00_foxlakecluster.yaml
+else ifeq ($(TEST_TYPE), oceanbase)
+ $(HELM) dependency build deploy/oceanbase-cluster --skip-refresh
+ $(HELM) upgrade --install oceanbase deploy/oceanbase
+ $(HELM) template oceanbase-cluster deploy/oceanbase-cluster > test/e2e/testdata/smoketest/oceanbase/00_oceanbasecluster.yaml
+else ifeq ($(TEST_TYPE), official-postgresql)
+ $(HELM) dependency build deploy/official-postgresql-cluster --skip-refresh
+ $(HELM) upgrade --install official-postgresql deploy/official-postgresql
+ $(HELM) template official-pg deploy/official-postgresql-cluster > test/e2e/testdata/smoketest/official-postgresql/00_official_pgcluster.yaml
+else ifeq ($(TEST_TYPE), openldap)
+ $(HELM) dependency build deploy/openldap-cluster --skip-refresh
+ $(HELM) upgrade --install openldap deploy/openldap
+ $(HELM) template openldap-cluster deploy/openldap-cluster > test/e2e/testdata/smoketest/openldap/00_openldapcluster.yaml
+else ifeq ($(TEST_TYPE), orioledb)
+ $(HELM) dependency build deploy/orioledb-cluster --skip-refresh
+ $(HELM) upgrade --install orioledb deploy/orioledb
+ $(HELM) template oriole-cluster deploy/orioledb-cluster > test/e2e/testdata/smoketest/orioledb/00_orioledbcluster.yaml
else
$(error "test type does not exist")
endif
@@ -685,6 +701,12 @@ else ifeq ($(TEST_TYPE), kafka)
$(HELM) upgrade --install kafka deploy/kafka
else ifeq ($(TEST_TYPE), foxlake)
$(HELM) upgrade --install foxlake deploy/foxlake
+else ifeq ($(TEST_TYPE), oceanbase)
+ $(HELM) upgrade --install oceanbase deploy/oceanbase
+else ifeq ($(TEST_TYPE), oceanbase)
+ $(HELM) upgrade --install official-postgresql deploy/official-postgresql
+else ifeq ($(TEST_TYPE), openldap)
+ $(HELM) upgrade --install openldap deploy/openldap
else
$(error "test type does not exist")
endif
diff --git a/PROJECT b/PROJECT
index 81732375d10..807651ecd19 100644
--- a/PROJECT
+++ b/PROJECT
@@ -39,8 +39,8 @@ resources:
namespaced: true
controller: true
domain: kubeblocks.io
- group: apps
- kind: BackupTool
+ group: dataprotection
+ kind: ActionSet
path: github.com/apecloud/kubeblocks/apis/dataprotection/v1alpha1
version: v1alpha1
- api:
@@ -66,18 +66,9 @@ resources:
namespaced: true
controller: true
domain: kubeblocks.io
- group: dataprotection
- kind: RestoreJob
- path: github.com/apecloud/kubeblocks/apis/dataprotection/v1alpha1
- version: v1alpha1
-- api:
- crdVersion: v1
- namespaced: true
- controller: true
- domain: kubeblocks.io
- group: dataprotection
+ group: apps
kind: BackupPolicyTemplate
- path: github.com/apecloud/kubeblocks/apis/dataprotection/v1alpha1
+ path: github.com/apecloud/kubeblocks/apis/apps/v1alpha1
version: v1alpha1
- api:
crdVersion: v1
@@ -168,6 +159,14 @@ resources:
kind: BackupRepo
path: github.com/apecloud/kubeblocks/apis/dataprotection/v1alpha1
version: v1alpha1
+- api:
+ crdVersion: v1
+ controller: true
+ domain: kubeblocks.io
+ group: dataprotection
+ kind: BackupSchedule
+ path: github.com/apecloud/kubeblocks/apis/dataprotection/v1alpha1
+ version: v1alpha1
- api:
crdVersion: v1
namespaced: true
@@ -190,4 +189,13 @@ resources:
defaulting: true
validation: true
webhookVersion: v1
+- api:
+ crdVersion: v1
+ namespaced: true
+ controller: true
+ domain: kubeblocks.io
+ group: dataprotection
+ kind: Restore
+ path: github.com/apecloud/kubeblocks/apis/dataprotection/v1alpha1
+ version: v1alpha1
version: "3"
diff --git a/README.md b/README.md
index ec565b3572a..ebd02955959 100644
--- a/README.md
+++ b/README.md
@@ -69,7 +69,7 @@ When adopting a multi-cloud or hybrid cloud strategy, it is essential to priorit
## Community
-- KubeBlocks [Slack Channel](https://kubeblocks.slack.com/join/shared_invite/zt-22cx5p0y9-~BDNuPqxkdgswI_FSdx_8g)
+- KubeBlocks [Slack Channel](https://join.slack.com/t/kubeblocks/shared_invite/zt-23vym7xpx-Xu3xcE7HmcqGKvTX4U9yTg)
- KubeBlocks Github [Discussions](https://github.com/apecloud/kubeblocks/discussions)
## Contributing to KubeBlocks
diff --git a/apis/apps/v1alpha1/backuppolicytemplate_types.go b/apis/apps/v1alpha1/backuppolicytemplate_types.go
index d7e61f71e7e..34431884905 100644
--- a/apis/apps/v1alpha1/backuppolicytemplate_types.go
+++ b/apis/apps/v1alpha1/backuppolicytemplate_types.go
@@ -18,6 +18,8 @@ package v1alpha1
import (
metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
+
+ dpv1alpha1 "github.com/apecloud/kubeblocks/apis/dataprotection/v1alpha1"
)
// BackupPolicyTemplateSpec defines the desired state of BackupPolicyTemplate
@@ -54,101 +56,46 @@ type BackupPolicy struct {
// +kubebuilder:validation:Pattern:=`^[a-z]([a-z0-9\-]*[a-z0-9])?$`
ComponentDefRef string `json:"componentDefRef"`
- // retention describe how long the Backup should be retained. if not set, will be retained forever.
- // +optional
- Retention *RetentionSpec `json:"retention,omitempty"`
-
- // schedule policy for backup.
- // +optional
- Schedule Schedule `json:"schedule,omitempty"`
-
- // the policy for snapshot backup.
- // +optional
- Snapshot *SnapshotPolicy `json:"snapshot,omitempty"`
-
- // the policy for datafile backup.
- // +optional
- Datafile *CommonBackupPolicy `json:"datafile,omitempty"`
-
- // the policy for logfile backup.
- // +optional
- Logfile *CommonBackupPolicy `json:"logfile,omitempty"`
-}
-
-type RetentionSpec struct {
- // ttl is a time string ending with the 'd'|'D'|'h'|'H' character to describe how long
- // the Backup should be retained. if not set, will be retained forever.
- // +kubebuilder:validation:Pattern:=`^\d+[d|D|h|H]$`
- // +optional
- TTL *string `json:"ttl,omitempty"`
-}
-
-type Schedule struct {
- // startingDeadlineMinutes defines the deadline in minutes for starting the backup job
- // if it misses scheduled time for any reason.
+ // retentionPeriod determines a duration up to which the backup should be kept.
+ // controller will remove all backups that are older than the RetentionPeriod.
+ // For example, RetentionPeriod of `30d` will keep only the backups of last 30 days.
+ // Sample duration format:
+ // - years: 2y
+ // - months: 6mo
+ // - days: 30d
+ // - hours: 12h
+ // - minutes: 30m
+ // You can also combine the above durations. For example: 30d12h30m
// +optional
- // +kubebuilder:validation:Minimum=0
- // +kubebuilder:validation:Maximum=1440
- StartingDeadlineMinutes *int64 `json:"startingDeadlineMinutes,omitempty"`
+ // +kubebuilder:default="7d"
+ RetentionPeriod dpv1alpha1.RetentionPeriod `json:"retentionPeriod,omitempty"`
- // schedule policy for snapshot backup.
- // +optional
- Snapshot *SchedulePolicy `json:"snapshot,omitempty"`
-
- // schedule policy for datafile backup.
+ // target instance for backup.
// +optional
- Datafile *SchedulePolicy `json:"datafile,omitempty"`
+ Target TargetInstance `json:"target"`
- // schedule policy for logfile backup.
+ // schedule policy for backup.
// +optional
- Logfile *SchedulePolicy `json:"logfile,omitempty"`
-}
+ Schedules []SchedulePolicy `json:"schedules,omitempty"`
-type SchedulePolicy struct {
- // the cron expression for schedule, the timezone is in UTC. see https://en.wikipedia.org/wiki/Cron.
+ // backupMethods defines the backup methods.
// +kubebuilder:validation:Required
- CronExpression string `json:"cronExpression"`
-
- // enable or disable the schedule.
- // +kubebuilder:validation:Required
- Enable bool `json:"enable"`
+ BackupMethods []dpv1alpha1.BackupMethod `json:"backupMethods"`
}
-type SnapshotPolicy struct {
- BasePolicy `json:",inline"`
-
- // execute hook commands for backup.
+type SchedulePolicy struct {
+ // enabled specifies whether the backup schedule is enabled or not.
// +optional
- Hooks *BackupPolicyHook `json:"hooks,omitempty"`
-}
-
-type CommonBackupPolicy struct {
- BasePolicy `json:",inline"`
+ Enabled *bool `json:"enabled,omitempty"`
- // which backup tool to perform database backup, only support one tool.
+ // backupMethod specifies the backup method name that is defined in backupPolicy.
// +kubebuilder:validation:Required
- // +kubebuilder:validation:Pattern:=`^[a-z0-9]([a-z0-9\.\-]*[a-z0-9])?$`
- BackupToolName string `json:"backupToolName,omitempty"`
-}
-
-type BasePolicy struct {
- // target instance for backup.
- // +optional
- Target TargetInstance `json:"target"`
-
- // the number of automatic backups to retain. Value must be non-negative integer.
- // 0 means NO limit on the number of backups.
- // +kubebuilder:default=7
- // +optional
- BackupsHistoryLimit int32 `json:"backupsHistoryLimit,omitempty"`
+ BackupMethod string `json:"backupMethod"`
- // count of backup stop retries on fail.
- // +optional
- OnFailAttempted int32 `json:"onFailAttempted,omitempty"`
-
- // define how to update metadata for backup status.
- // +optional
- BackupStatusUpdates []BackupStatusUpdate `json:"backupStatusUpdates,omitempty"`
+ // the cron expression for schedule, the timezone is in UTC.
+ // see https://en.wikipedia.org/wiki/Cron.
+ // +kubebuilder:validation:Required
+ CronExpression string `json:"cronExpression"`
}
type TargetInstance struct {
@@ -185,51 +132,13 @@ type ConnectionCredentialKey struct {
// if not set, the default key is "username".
// +optional
UsernameKey *string `json:"usernameKey,omitempty"`
-}
-
-// BackupPolicyHook defines for the database execute commands before and after backup.
-type BackupPolicyHook struct {
- // pre backup to perform commands
- // +optional
- PreCommands []string `json:"preCommands,omitempty"`
- // post backup to perform commands
- // +optional
- PostCommands []string `json:"postCommands,omitempty"`
-
- // exec command with image
- // +optional
- Image string `json:"image,omitempty"`
-
- // which container can exec command
- // +optional
- ContainerName string `json:"containerName,omitempty"`
-}
-
-type BackupStatusUpdate struct {
- // specify the json path of backup object for patch.
- // example: manifests.backupLog -- means patch the backup json path of status.manifests.backupLog.
- // +optional
- Path string `json:"path,omitempty"`
+ // hostKey specifies the map key of the host in the connection credential secret.
+ HostKey *string `json:"hostKey,omitempty"`
- // which container name that kubectl can execute.
- // +optional
- ContainerName string `json:"containerName,omitempty"`
-
- // the shell Script commands to collect backup status metadata.
- // The script must exist in the container of ContainerName and the output format must be set to JSON.
- // Note that outputting to stderr may cause the result format to not be in JSON.
- // +optional
- Script string `json:"script,omitempty"`
-
- // useTargetPodServiceAccount defines whether this job requires the service account of the backup target pod.
- // if true, will use the service account of the backup target pod. otherwise, will use the system service account.
- // +optional
- UseTargetPodServiceAccount bool `json:"useTargetPodServiceAccount,omitempty"`
-
- // when to update the backup status, pre: before backup, post: after backup
- // +kubebuilder:validation:Required
- UpdateStage BackupStatusUpdateStage `json:"updateStage"`
+ // portKey specifies the map key of the port in the connection credential secret.
+ // +kubebuilder:default=port
+ PortKey *string `json:"portKey,omitempty"`
}
// BackupPolicyTemplateStatus defines the observed state of BackupPolicyTemplate
diff --git a/apis/apps/v1alpha1/cluster_types.go b/apis/apps/v1alpha1/cluster_types.go
index ccf65cc8507..f6ff22c0b8b 100644
--- a/apis/apps/v1alpha1/cluster_types.go
+++ b/apis/apps/v1alpha1/cluster_types.go
@@ -28,7 +28,8 @@ import (
"k8s.io/apimachinery/pkg/util/intstr"
"sigs.k8s.io/controller-runtime/pkg/client"
- dataprotectionv1alpha1 "github.com/apecloud/kubeblocks/apis/dataprotection/v1alpha1"
+ dpv1alpha1 "github.com/apecloud/kubeblocks/apis/dataprotection/v1alpha1"
+ workloads "github.com/apecloud/kubeblocks/apis/workloads/v1alpha1"
"github.com/apecloud/kubeblocks/internal/constant"
viper "github.com/apecloud/kubeblocks/internal/viperx"
)
@@ -115,18 +116,23 @@ type ClusterBackup struct {
// +optional
Enabled *bool `json:"enabled,omitempty"`
- // retentionPeriod is a time string ending with the 'd'|'D'|'h'|'H' character to describe how long
- // the Backup should be retained. if not set, will be retained forever.
- // +kubebuilder:validation:Pattern:=`^\d+[d|D|h|H]$`
- // +kubebuilder:default="1d"
+ // retentionPeriod determines a duration up to which the backup should be kept.
+ // controller will remove all backups that are older than the RetentionPeriod.
+ // For example, RetentionPeriod of `30d` will keep only the backups of last 30 days.
+ // Sample duration format:
+ // - years: 2y
+ // - months: 6mo
+ // - days: 30d
+ // - hours: 12h
+ // - minutes: 30m
+ // You can also combine the above durations. For example: 30d12h30m
+ // +kubebuilder:default="7d"
// +optional
- RetentionPeriod *string `json:"retentionPeriod,omitempty"`
+ RetentionPeriod dpv1alpha1.RetentionPeriod `json:"retentionPeriod,omitempty"`
- // backup method, support: snapshot, backupTool.
- // +kubebuilder:validation:Enum=snapshot;backupTool
- // +kubebuilder:validation:Required
- // +kubebuilder:default=snapshot
- Method dataprotectionv1alpha1.BackupMethod `json:"method"`
+ // backup method name to use, that is defined in backupPolicy.
+ // +optional
+ Method string `json:"method"`
// the cron expression for schedule, the timezone is in UTC. see https://en.wikipedia.org/wiki/Cron.
// +optional
@@ -134,9 +140,9 @@ type ClusterBackup struct {
// startingDeadlineMinutes defines the deadline in minutes for starting the backup job
// if it misses scheduled time for any reason.
- // +optional
// +kubebuilder:validation:Minimum=0
// +kubebuilder:validation:Maximum=1440
+ // +optional
StartingDeadlineMinutes *int64 `json:"startingDeadlineMinutes,omitempty"`
// repoName is the name of the backupRepo, if not set, will use the default backupRepo.
@@ -349,11 +355,17 @@ type ClusterComponentStatus struct {
// consensusSetStatus specifies the mapping of role and pod name.
// +optional
+ // +kubebuilder:deprecatedversion:warning="This field is deprecated from KB 0.7.0, use MembersStatus instead."
ConsensusSetStatus *ConsensusSetStatus `json:"consensusSetStatus,omitempty"`
// replicationSetStatus specifies the mapping of role and pod name.
// +optional
+ // +kubebuilder:deprecatedversion:warning="This field is deprecated from KB 0.7.0, use MembersStatus instead."
ReplicationSetStatus *ReplicationSetStatus `json:"replicationSetStatus,omitempty"`
+
+ // members' status.
+ // +optional
+ MembersStatus []workloads.MemberStatus `json:"membersStatus,omitempty"`
}
type ConsensusSetStatus struct {
diff --git a/apis/apps/v1alpha1/cluster_webhook.go b/apis/apps/v1alpha1/cluster_webhook.go
index df18f4e1318..471afad604c 100644
--- a/apis/apps/v1alpha1/cluster_webhook.go
+++ b/apis/apps/v1alpha1/cluster_webhook.go
@@ -30,6 +30,7 @@ import (
ctrl "sigs.k8s.io/controller-runtime"
logf "sigs.k8s.io/controller-runtime/pkg/log"
"sigs.k8s.io/controller-runtime/pkg/webhook"
+ "sigs.k8s.io/controller-runtime/pkg/webhook/admission"
)
// log is for logging in this package.
@@ -47,31 +48,31 @@ func (r *Cluster) SetupWebhookWithManager(mgr ctrl.Manager) error {
var _ webhook.Validator = &Cluster{}
// ValidateCreate implements webhook.Validator so a webhook will be registered for the type
-func (r *Cluster) ValidateCreate() error {
+func (r *Cluster) ValidateCreate() (admission.Warnings, error) {
clusterlog.Info("validate create", "name", r.Name)
- return r.validate()
+ return nil, r.validate()
}
// ValidateUpdate implements webhook.Validator so a webhook will be registered for the type
-func (r *Cluster) ValidateUpdate(old runtime.Object) error {
+func (r *Cluster) ValidateUpdate(old runtime.Object) (admission.Warnings, error) {
clusterlog.Info("validate update", "name", r.Name)
lastCluster := old.(*Cluster)
if lastCluster.Spec.ClusterDefRef != r.Spec.ClusterDefRef {
- return newInvalidError(ClusterKind, r.Name, "spec.clusterDefinitionRef", "clusterDefinitionRef is immutable, you can not update it. ")
+ return nil, newInvalidError(ClusterKind, r.Name, "spec.clusterDefinitionRef", "clusterDefinitionRef is immutable, you can not update it. ")
}
if err := r.validate(); err != nil {
- return err
+ return nil, err
}
- return r.validateVolumeClaimTemplates(lastCluster)
+ return nil, r.validateVolumeClaimTemplates(lastCluster)
}
// ValidateDelete implements webhook.Validator so a webhook will be registered for the type
-func (r *Cluster) ValidateDelete() error {
+func (r *Cluster) ValidateDelete() (admission.Warnings, error) {
clusterlog.Info("validate delete", "name", r.Name)
if r.Spec.TerminationPolicy == DoNotTerminate {
- return fmt.Errorf("the deletion for a cluster with DoNotTerminate termination policy is denied")
+ return nil, fmt.Errorf("the deletion for a cluster with DoNotTerminate termination policy is denied")
}
- return nil
+ return nil, nil
}
// validateVolumeClaimTemplates volumeClaimTemplates is forbidden modification except for storage size.
diff --git a/apis/apps/v1alpha1/clusterdefinition_types.go b/apis/apps/v1alpha1/clusterdefinition_types.go
index 141ed5cc8c8..63b66e095cf 100644
--- a/apis/apps/v1alpha1/clusterdefinition_types.go
+++ b/apis/apps/v1alpha1/clusterdefinition_types.go
@@ -23,6 +23,8 @@ import (
corev1 "k8s.io/api/core/v1"
metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
"k8s.io/apimachinery/pkg/util/intstr"
+
+ workloads "github.com/apecloud/kubeblocks/apis/workloads/v1alpha1"
)
// ClusterDefinitionSpec defines the desired state of ClusterDefinition
@@ -303,7 +305,7 @@ type ServiceRefDeclarationSpec struct {
// ClusterComponentDefinition provides a workload component specification template,
// with attributes that strongly work with stateful workloads and day-2 operations
// behaviors.
-// +kubebuilder:validation:XValidation:rule="has(self.workloadType) && self.workloadType == 'Consensus' ? has(self.consensusSpec) : !has(self.consensusSpec)",message="componentDefs.consensusSpec is required when componentDefs.workloadType is Consensus, and forbidden otherwise"
+// +kubebuilder:validation:XValidation:rule="has(self.workloadType) && self.workloadType == 'Consensus' ? (has(self.consensusSpec) || has(self.rsmSpec)) : !has(self.consensusSpec)",message="componentDefs.consensusSpec(deprecated) or componentDefs.rsmSpec(recommended) is required when componentDefs.workloadType is Consensus, and forbidden otherwise"
type ClusterComponentDefinition struct {
// A component definition name, this name could be used as default name of `Cluster.spec.componentSpecs.name`,
// and so this name is need to conform with same validation rules as `Cluster.spec.componentSpecs.name`, that
@@ -378,20 +380,30 @@ type ClusterComponentDefinition struct {
// statelessSpec defines stateless related spec if workloadType is Stateless.
// +optional
+ //+kubebuilder:deprecatedversion:warning="This field is deprecated from KB 0.7.0, use RSMSpec instead."
StatelessSpec *StatelessSetSpec `json:"statelessSpec,omitempty"`
// statefulSpec defines stateful related spec if workloadType is Stateful.
// +optional
+ //+kubebuilder:deprecatedversion:warning="This field is deprecated from KB 0.7.0, use RSMSpec instead."
StatefulSpec *StatefulSetSpec `json:"statefulSpec,omitempty"`
// consensusSpec defines consensus related spec if workloadType is Consensus, required if workloadType is Consensus.
// +optional
+ //+kubebuilder:deprecatedversion:warning="This field is deprecated from KB 0.7.0, use RSMSpec instead."
ConsensusSpec *ConsensusSetSpec `json:"consensusSpec,omitempty"`
// replicationSpec defines replication related spec if workloadType is Replication.
// +optional
+ //+kubebuilder:deprecatedversion:warning="This field is deprecated from KB 0.7.0, use RSMSpec instead."
ReplicationSpec *ReplicationSetSpec `json:"replicationSpec,omitempty"`
+ // RSMSpec defines workload related spec of this component.
+ // start from KB 0.7.0, RSM(ReplicatedStateMachineSpec) will be the underlying CR which powers all kinds of workload in KB.
+ // RSM is an enhanced stateful workload extension dedicated for heavy-state workloads like databases.
+ // +optional
+ RSMSpec *RSMSpec `json:"rsmSpec,omitempty"`
+
// horizontalScalePolicy controls the behavior of horizontal scale.
// +optional
HorizontalScalePolicy *HorizontalScalePolicy `json:"horizontalScalePolicy,omitempty"`
@@ -701,6 +713,7 @@ type ClusterDefinitionProbes struct {
// Probe for DB role changed check.
// +optional
+ //+kubebuilder:deprecatedversion:warning="This field is deprecated from KB 0.7.0, use RSMSpec instead."
RoleProbe *ClusterDefinitionProbe `json:"roleProbe,omitempty"`
// roleProbeTimeoutAfterPodsReady(in seconds), when all pods of the component are ready,
@@ -881,6 +894,30 @@ type ConsensusMember struct {
Replicas *int32 `json:"replicas,omitempty"`
}
+type RSMSpec struct {
+ // Roles, a list of roles defined in the system.
+ // +optional
+ Roles []workloads.ReplicaRole `json:"roles,omitempty"`
+
+ // RoleProbe provides method to probe role.
+ // +optional
+ RoleProbe *workloads.RoleProbe `json:"roleProbe,omitempty"`
+
+ // MembershipReconfiguration provides actions to do membership dynamic reconfiguration.
+ // +optional
+ MembershipReconfiguration *workloads.MembershipReconfiguration `json:"membershipReconfiguration,omitempty"`
+
+ // MemberUpdateStrategy, Members(Pods) update strategy.
+ // serial: update Members one by one that guarantee minimum component unavailable time.
+ // Learner -> Follower(with AccessMode=none) -> Follower(with AccessMode=readonly) -> Follower(with AccessMode=readWrite) -> Leader
+ // bestEffortParallel: update Members in parallel that guarantee minimum component un-writable time.
+ // Learner, Follower(minority) in parallel -> Follower(majority) -> Leader, keep majority online all the time.
+ // parallel: force parallel
+ // +kubebuilder:validation:Enum={Serial,BestEffortParallel,Parallel}
+ // +optional
+ MemberUpdateStrategy *workloads.MemberUpdateStrategy `json:"memberUpdateStrategy,omitempty"`
+}
+
type ReplicationSetSpec struct {
StatefulSetSpec `json:",inline"`
}
diff --git a/apis/apps/v1alpha1/clusterdefinition_webhook.go b/apis/apps/v1alpha1/clusterdefinition_webhook.go
index 6e3ef387fe8..f5290a581e4 100644
--- a/apis/apps/v1alpha1/clusterdefinition_webhook.go
+++ b/apis/apps/v1alpha1/clusterdefinition_webhook.go
@@ -28,6 +28,7 @@ import (
ctrl "sigs.k8s.io/controller-runtime"
logf "sigs.k8s.io/controller-runtime/pkg/log"
"sigs.k8s.io/controller-runtime/pkg/webhook"
+ "sigs.k8s.io/controller-runtime/pkg/webhook/admission"
)
// log is for logging in this package.
@@ -82,21 +83,21 @@ func (r *ClusterDefinition) Default() {
var _ webhook.Validator = &ClusterDefinition{}
// ValidateCreate implements webhook.Validator so a webhook will be registered for the type
-func (r *ClusterDefinition) ValidateCreate() error {
+func (r *ClusterDefinition) ValidateCreate() (admission.Warnings, error) {
clusterdefinitionlog.Info("validate create", "name", r.Name)
- return r.validate()
+ return nil, r.validate()
}
// ValidateUpdate implements webhook.Validator so a webhook will be registered for the type
-func (r *ClusterDefinition) ValidateUpdate(old runtime.Object) error {
+func (r *ClusterDefinition) ValidateUpdate(old runtime.Object) (admission.Warnings, error) {
clusterdefinitionlog.Info("validate update", "name", r.Name)
- return r.validate()
+ return nil, r.validate()
}
// ValidateDelete implements webhook.Validator so a webhook will be registered for the type
-func (r *ClusterDefinition) ValidateDelete() error {
+func (r *ClusterDefinition) ValidateDelete() (admission.Warnings, error) {
clusterdefinitionlog.Info("validate delete", "name", r.Name)
- return nil
+ return nil, nil
}
// Validate ClusterDefinition.spec is legal
diff --git a/apis/apps/v1alpha1/clusterversion_webhook.go b/apis/apps/v1alpha1/clusterversion_webhook.go
index 16c37b92b30..ba2727fba4f 100644
--- a/apis/apps/v1alpha1/clusterversion_webhook.go
+++ b/apis/apps/v1alpha1/clusterversion_webhook.go
@@ -29,6 +29,7 @@ import (
ctrl "sigs.k8s.io/controller-runtime"
logf "sigs.k8s.io/controller-runtime/pkg/log"
"sigs.k8s.io/controller-runtime/pkg/webhook"
+ "sigs.k8s.io/controller-runtime/pkg/webhook/admission"
)
// log is for logging in this package.
@@ -46,26 +47,26 @@ func (r *ClusterVersion) SetupWebhookWithManager(mgr ctrl.Manager) error {
var _ webhook.Validator = &ClusterVersion{}
// ValidateCreate implements webhook.Validator so a webhook will be registered for the type
-func (r *ClusterVersion) ValidateCreate() error {
+func (r *ClusterVersion) ValidateCreate() (admission.Warnings, error) {
clusterversionlog.Info("validate create", "name", r.Name)
- return r.validate()
+ return nil, r.validate()
}
// ValidateUpdate implements webhook.Validator so a webhook will be registered for the type
-func (r *ClusterVersion) ValidateUpdate(old runtime.Object) error {
+func (r *ClusterVersion) ValidateUpdate(old runtime.Object) (admission.Warnings, error) {
clusterversionlog.Info("validate update", "name", r.Name)
// determine whether r.spec content is modified
lastClusterVersion := old.(*ClusterVersion)
if !reflect.DeepEqual(lastClusterVersion.Spec, r.Spec) {
- return newInvalidError(ClusterVersionKind, r.Name, "", "ClusterVersion.spec is immutable, you can not update it.")
+ return nil, newInvalidError(ClusterVersionKind, r.Name, "", "ClusterVersion.spec is immutable, you can not update it.")
}
- return nil
+ return nil, nil
}
// ValidateDelete implements webhook.Validator so a webhook will be registered for the type
-func (r *ClusterVersion) ValidateDelete() error {
+func (r *ClusterVersion) ValidateDelete() (admission.Warnings, error) {
clusterversionlog.Info("validate delete", "name", r.Name)
- return nil
+ return nil, nil
}
// Validate ClusterVersion.spec is legal
diff --git a/apis/apps/v1alpha1/config.go b/apis/apps/v1alpha1/config.go
index bc91687dd17..c0cf6afbe1c 100644
--- a/apis/apps/v1alpha1/config.go
+++ b/apis/apps/v1alpha1/config.go
@@ -72,10 +72,11 @@ package v1alpha1
// ConfigurationPhase defines the Configuration FSM phase
// +enum
-// +kubebuilder:validation:Enum={Init,Running,Pending,Merged,MergeFailed,FailedAndPause,Upgrading,Deleting,FailedAndRetry,Finished}
+// +kubebuilder:validation:Enum={Creating,Init,Running,Pending,Merged,MergeFailed,FailedAndPause,Upgrading,Deleting,FailedAndRetry,Finished}
type ConfigurationPhase string
const (
+ CCreatingPhase ConfigurationPhase = "Creating"
CInitPhase ConfigurationPhase = "Init"
CRunningPhase ConfigurationPhase = "Running"
CPendingPhase ConfigurationPhase = "Pending"
diff --git a/apis/apps/v1alpha1/configuration_types.go b/apis/apps/v1alpha1/configuration_types.go
index 3699434ad4d..272f0f0fd99 100644
--- a/apis/apps/v1alpha1/configuration_types.go
+++ b/apis/apps/v1alpha1/configuration_types.go
@@ -37,9 +37,13 @@ type ConfigurationItemDetail struct {
// +optional
Version string `json:"version,omitempty"`
+ // configSpec is used to set the configuration template.
+ // +optional
+ ConfigSpec *ComponentConfigSpec `json:"configSpec"`
+
// Specify the configuration template.
// +optional
- ImportTemplateRef *ConfigTemplateExtension `json:"importTemplateRef,omitempty"`
+ ImportTemplateRef *ConfigTemplateExtension `json:"importTemplateRef"`
// configFileParams is used to set the parameters to be updated.
// +optional
@@ -61,16 +65,6 @@ type ConfigurationSpec struct {
// +kubebuilder:validation:XValidation:rule="self == oldSelf",message="forbidden to update spec.clusterRef"
ComponentName string `json:"componentName"`
- // clusterDefRef referencing ClusterDefinition name. This is an immutable attribute.
- // +kubebuilder:validation:Required
- // +kubebuilder:validation:Pattern:=`^[a-z0-9]([a-z0-9\.\-]*[a-z0-9])?$`
- ClusterDefRef string `json:"clusterDefRef"`
-
- // clusterVerRef referencing ClusterVersion name.
- // +kubebuilder:validation:Pattern:=`^[a-z0-9]([a-z0-9\.\-]*[a-z0-9])?$`
- // +optional
- ClusterVersionRef string `json:"clusterVerRef,omitempty"`
-
// customConfigurationItems describes user-defined config template.
// +optional
// +patchMergeKey=name
diff --git a/apis/apps/v1alpha1/opsrequest_types.go b/apis/apps/v1alpha1/opsrequest_types.go
index 1293d70b680..20af28d8e59 100644
--- a/apis/apps/v1alpha1/opsrequest_types.go
+++ b/apis/apps/v1alpha1/opsrequest_types.go
@@ -354,6 +354,13 @@ type ScriptSpec struct {
// +optional
// +kubebuilder:validation:XValidation:rule="self == oldSelf",message="forbidden to update spec.scriptSpec.scriptFrom"
ScriptFrom *ScriptFrom `json:"scriptFrom,omitempty"`
+ // KubeBlocks, by default, will execute the script on the primary pod, with role=leader.
+ // There are some exceptions, such as Redis, which does not synchronize accounts info between primary and secondary.
+ // In this case, we need to execute the script on all pods, matching the selector.
+ // selector indicates the components on which the script is executed.
+ // +optional
+ // +kubebuilder:validation:XValidation:rule="self == oldSelf",message="forbidden to update spec.scriptSpec.script.selector"
+ Selector *metav1.LabelSelector `json:"selector,omitempty"`
}
type BackupSpec struct {
@@ -365,11 +372,9 @@ type BackupSpec struct {
// +optional
BackupPolicyName string `json:"backupPolicyName"`
- // Backup Type. datafile or logfile or snapshot. If not set, datafile is the default type.
- // +kubebuilder:default=datafile
- // +kubeBuilder:validation:Enum={datafile,logfile,snapshot}
+ // Backup method name that is defined in backupPolicy.
// +optional
- BackupType string `json:"backupType"`
+ BackupMethod string `json:"backupMethod"`
// if backupType is incremental, parentBackupName is required.
// +optional
diff --git a/apis/apps/v1alpha1/opsrequest_webhook.go b/apis/apps/v1alpha1/opsrequest_webhook.go
index 7655db8f903..a345a2f4592 100644
--- a/apis/apps/v1alpha1/opsrequest_webhook.go
+++ b/apis/apps/v1alpha1/opsrequest_webhook.go
@@ -35,6 +35,7 @@ import (
"sigs.k8s.io/controller-runtime/pkg/client"
logf "sigs.k8s.io/controller-runtime/pkg/log"
"sigs.k8s.io/controller-runtime/pkg/webhook"
+ "sigs.k8s.io/controller-runtime/pkg/webhook/admission"
"github.com/apecloud/kubeblocks/internal/constant"
)
@@ -59,38 +60,38 @@ func (r *OpsRequest) SetupWebhookWithManager(mgr ctrl.Manager) error {
var _ webhook.Validator = &OpsRequest{}
// ValidateCreate implements webhook.Validator so a webhook will be registered for the type
-func (r *OpsRequest) ValidateCreate() error {
+func (r *OpsRequest) ValidateCreate() (admission.Warnings, error) {
opsRequestLog.Info("validate create", "name", r.Name)
- return r.validateEntry(true)
+ return nil, r.validateEntry(true)
}
// ValidateUpdate implements webhook.Validator so a webhook will be registered for the type
-func (r *OpsRequest) ValidateUpdate(old runtime.Object) error {
+func (r *OpsRequest) ValidateUpdate(old runtime.Object) (admission.Warnings, error) {
opsRequestLog.Info("validate update", "name", r.Name)
lastOpsRequest := old.(*OpsRequest).DeepCopy()
// if no spec updated, we should skip validation.
// if not, we can not delete the OpsRequest when cluster has been deleted.
// because when cluster not existed, r.validate will report an error.
if reflect.DeepEqual(lastOpsRequest.Spec, r.Spec) {
- return nil
+ return nil, nil
}
if r.IsComplete() {
- return fmt.Errorf("update OpsRequest: %s is forbidden when status.Phase is %s", r.Name, r.Status.Phase)
+ return nil, fmt.Errorf("update OpsRequest: %s is forbidden when status.Phase is %s", r.Name, r.Status.Phase)
}
// Keep the cancel consistent between the two opsRequest for comparing the diff.
lastOpsRequest.Spec.Cancel = r.Spec.Cancel
if !reflect.DeepEqual(lastOpsRequest.Spec, r.Spec) && r.Status.Phase != "" {
- return fmt.Errorf("update OpsRequest: %s is forbidden except for cancel when status.Phase is %s", r.Name, r.Status.Phase)
+ return nil, fmt.Errorf("update OpsRequest: %s is forbidden except for cancel when status.Phase is %s", r.Name, r.Status.Phase)
}
- return r.validateEntry(false)
+ return nil, r.validateEntry(false)
}
// ValidateDelete implements webhook.Validator so a webhook will be registered for the type
-func (r *OpsRequest) ValidateDelete() error {
+func (r *OpsRequest) ValidateDelete() (admission.Warnings, error) {
opsRequestLog.Info("validate delete", "name", r.Name)
- return nil
+ return nil, nil
}
// IsComplete checks if opsRequest has been completed.
diff --git a/apis/apps/v1alpha1/servicedescriptor_webhook.go b/apis/apps/v1alpha1/servicedescriptor_webhook.go
index 84fc9caa630..4ca9f42e7af 100644
--- a/apis/apps/v1alpha1/servicedescriptor_webhook.go
+++ b/apis/apps/v1alpha1/servicedescriptor_webhook.go
@@ -27,6 +27,7 @@ import (
ctrl "sigs.k8s.io/controller-runtime"
logf "sigs.k8s.io/controller-runtime/pkg/log"
"sigs.k8s.io/controller-runtime/pkg/webhook"
+ "sigs.k8s.io/controller-runtime/pkg/webhook/admission"
)
// log is for logging in this package.
@@ -57,24 +58,24 @@ func (r *ServiceDescriptor) Default() {
var _ webhook.Validator = &ServiceDescriptor{}
// ValidateCreate implements webhook.Validator so a webhook will be registered for the type
-func (r *ServiceDescriptor) ValidateCreate() error {
+func (r *ServiceDescriptor) ValidateCreate() (admission.Warnings, error) {
servicedescriptorlog.Info("validate create", "name", r.Name)
- return r.validate()
+ return nil, r.validate()
}
// ValidateUpdate implements webhook.Validator so a webhook will be registered for the type
-func (r *ServiceDescriptor) ValidateUpdate(old runtime.Object) error {
+func (r *ServiceDescriptor) ValidateUpdate(old runtime.Object) (admission.Warnings, error) {
servicedescriptorlog.Info("validate update", "name", r.Name)
- return r.validate()
+ return nil, r.validate()
}
// ValidateDelete implements webhook.Validator so a webhook will be registered for the type
-func (r *ServiceDescriptor) ValidateDelete() error {
+func (r *ServiceDescriptor) ValidateDelete() (admission.Warnings, error) {
servicedescriptorlog.Info("validate delete", "name", r.Name)
- return r.validate()
+ return nil, r.validate()
}
func (r *ServiceDescriptor) validate() error {
diff --git a/apis/apps/v1alpha1/zz_generated.deepcopy.go b/apis/apps/v1alpha1/zz_generated.deepcopy.go
index 4a5aa7365f1..fa5956b72f4 100644
--- a/apis/apps/v1alpha1/zz_generated.deepcopy.go
+++ b/apis/apps/v1alpha1/zz_generated.deepcopy.go
@@ -25,6 +25,8 @@ along with this program. If not, see .
package v1alpha1
import (
+ dataprotectionv1alpha1 "github.com/apecloud/kubeblocks/apis/dataprotection/v1alpha1"
+ workloadsv1alpha1 "github.com/apecloud/kubeblocks/apis/workloads/v1alpha1"
appsv1 "k8s.io/api/apps/v1"
"k8s.io/api/core/v1"
"k8s.io/apimachinery/pkg/api/resource"
@@ -63,26 +65,20 @@ func (in *Affinity) DeepCopy() *Affinity {
// DeepCopyInto is an autogenerated deepcopy function, copying the receiver, writing into out. in must be non-nil.
func (in *BackupPolicy) DeepCopyInto(out *BackupPolicy) {
*out = *in
- if in.Retention != nil {
- in, out := &in.Retention, &out.Retention
- *out = new(RetentionSpec)
- (*in).DeepCopyInto(*out)
- }
- in.Schedule.DeepCopyInto(&out.Schedule)
- if in.Snapshot != nil {
- in, out := &in.Snapshot, &out.Snapshot
- *out = new(SnapshotPolicy)
- (*in).DeepCopyInto(*out)
- }
- if in.Datafile != nil {
- in, out := &in.Datafile, &out.Datafile
- *out = new(CommonBackupPolicy)
- (*in).DeepCopyInto(*out)
+ in.Target.DeepCopyInto(&out.Target)
+ if in.Schedules != nil {
+ in, out := &in.Schedules, &out.Schedules
+ *out = make([]SchedulePolicy, len(*in))
+ for i := range *in {
+ (*in)[i].DeepCopyInto(&(*out)[i])
+ }
}
- if in.Logfile != nil {
- in, out := &in.Logfile, &out.Logfile
- *out = new(CommonBackupPolicy)
- (*in).DeepCopyInto(*out)
+ if in.BackupMethods != nil {
+ in, out := &in.BackupMethods, &out.BackupMethods
+ *out = make([]dataprotectionv1alpha1.BackupMethod, len(*in))
+ for i := range *in {
+ (*in)[i].DeepCopyInto(&(*out)[i])
+ }
}
}
@@ -96,31 +92,6 @@ func (in *BackupPolicy) DeepCopy() *BackupPolicy {
return out
}
-// DeepCopyInto is an autogenerated deepcopy function, copying the receiver, writing into out. in must be non-nil.
-func (in *BackupPolicyHook) DeepCopyInto(out *BackupPolicyHook) {
- *out = *in
- if in.PreCommands != nil {
- in, out := &in.PreCommands, &out.PreCommands
- *out = make([]string, len(*in))
- copy(*out, *in)
- }
- if in.PostCommands != nil {
- in, out := &in.PostCommands, &out.PostCommands
- *out = make([]string, len(*in))
- copy(*out, *in)
- }
-}
-
-// DeepCopy is an autogenerated deepcopy function, copying the receiver, creating a new BackupPolicyHook.
-func (in *BackupPolicyHook) DeepCopy() *BackupPolicyHook {
- if in == nil {
- return nil
- }
- out := new(BackupPolicyHook)
- in.DeepCopyInto(out)
- return out
-}
-
// DeepCopyInto is an autogenerated deepcopy function, copying the receiver, writing into out. in must be non-nil.
func (in *BackupPolicyTemplate) DeepCopyInto(out *BackupPolicyTemplate) {
*out = *in
@@ -248,42 +219,6 @@ func (in *BackupSpec) DeepCopy() *BackupSpec {
return out
}
-// DeepCopyInto is an autogenerated deepcopy function, copying the receiver, writing into out. in must be non-nil.
-func (in *BackupStatusUpdate) DeepCopyInto(out *BackupStatusUpdate) {
- *out = *in
-}
-
-// DeepCopy is an autogenerated deepcopy function, copying the receiver, creating a new BackupStatusUpdate.
-func (in *BackupStatusUpdate) DeepCopy() *BackupStatusUpdate {
- if in == nil {
- return nil
- }
- out := new(BackupStatusUpdate)
- in.DeepCopyInto(out)
- return out
-}
-
-// DeepCopyInto is an autogenerated deepcopy function, copying the receiver, writing into out. in must be non-nil.
-func (in *BasePolicy) DeepCopyInto(out *BasePolicy) {
- *out = *in
- in.Target.DeepCopyInto(&out.Target)
- if in.BackupStatusUpdates != nil {
- in, out := &in.BackupStatusUpdates, &out.BackupStatusUpdates
- *out = make([]BackupStatusUpdate, len(*in))
- copy(*out, *in)
- }
-}
-
-// DeepCopy is an autogenerated deepcopy function, copying the receiver, creating a new BasePolicy.
-func (in *BasePolicy) DeepCopy() *BasePolicy {
- if in == nil {
- return nil
- }
- out := new(BasePolicy)
- in.DeepCopyInto(out)
- return out
-}
-
// DeepCopyInto is an autogenerated deepcopy function, copying the receiver, writing into out. in must be non-nil.
func (in *CPUConstraint) DeepCopyInto(out *CPUConstraint) {
*out = *in
@@ -371,11 +306,6 @@ func (in *ClusterBackup) DeepCopyInto(out *ClusterBackup) {
*out = new(bool)
**out = **in
}
- if in.RetentionPeriod != nil {
- in, out := &in.RetentionPeriod, &out.RetentionPeriod
- *out = new(string)
- **out = **in
- }
if in.StartingDeadlineMinutes != nil {
in, out := &in.StartingDeadlineMinutes, &out.StartingDeadlineMinutes
*out = new(int64)
@@ -460,6 +390,11 @@ func (in *ClusterComponentDefinition) DeepCopyInto(out *ClusterComponentDefiniti
*out = new(ReplicationSetSpec)
(*in).DeepCopyInto(*out)
}
+ if in.RSMSpec != nil {
+ in, out := &in.RSMSpec, &out.RSMSpec
+ *out = new(RSMSpec)
+ (*in).DeepCopyInto(*out)
+ }
if in.HorizontalScalePolicy != nil {
in, out := &in.HorizontalScalePolicy, &out.HorizontalScalePolicy
*out = new(HorizontalScalePolicy)
@@ -636,6 +571,11 @@ func (in *ClusterComponentStatus) DeepCopyInto(out *ClusterComponentStatus) {
*out = new(ReplicationSetStatus)
(*in).DeepCopyInto(*out)
}
+ if in.MembersStatus != nil {
+ in, out := &in.MembersStatus, &out.MembersStatus
+ *out = make([]workloadsv1alpha1.MemberStatus, len(*in))
+ copy(*out, *in)
+ }
}
// DeepCopy is an autogenerated deepcopy function, copying the receiver, creating a new ClusterComponentStatus.
@@ -1253,22 +1193,6 @@ func (in *CommandExecutorItem) DeepCopy() *CommandExecutorItem {
return out
}
-// DeepCopyInto is an autogenerated deepcopy function, copying the receiver, writing into out. in must be non-nil.
-func (in *CommonBackupPolicy) DeepCopyInto(out *CommonBackupPolicy) {
- *out = *in
- in.BasePolicy.DeepCopyInto(&out.BasePolicy)
-}
-
-// DeepCopy is an autogenerated deepcopy function, copying the receiver, creating a new CommonBackupPolicy.
-func (in *CommonBackupPolicy) DeepCopy() *CommonBackupPolicy {
- if in == nil {
- return nil
- }
- out := new(CommonBackupPolicy)
- in.DeepCopyInto(out)
- return out
-}
-
// DeepCopyInto is an autogenerated deepcopy function, copying the receiver, writing into out. in must be non-nil.
func (in *ComponentClass) DeepCopyInto(out *ComponentClass) {
*out = *in
@@ -1963,6 +1887,11 @@ func (in *ConfigurationItem) DeepCopy() *ConfigurationItem {
// DeepCopyInto is an autogenerated deepcopy function, copying the receiver, writing into out. in must be non-nil.
func (in *ConfigurationItemDetail) DeepCopyInto(out *ConfigurationItemDetail) {
*out = *in
+ if in.ConfigSpec != nil {
+ in, out := &in.ConfigSpec, &out.ConfigSpec
+ *out = new(ComponentConfigSpec)
+ (*in).DeepCopyInto(*out)
+ }
if in.ImportTemplateRef != nil {
in, out := &in.ImportTemplateRef, &out.ImportTemplateRef
*out = new(ConfigTemplateExtension)
@@ -2151,6 +2080,16 @@ func (in *ConnectionCredentialKey) DeepCopyInto(out *ConnectionCredentialKey) {
*out = new(string)
**out = **in
}
+ if in.HostKey != nil {
+ in, out := &in.HostKey, &out.HostKey
+ *out = new(string)
+ **out = **in
+ }
+ if in.PortKey != nil {
+ in, out := &in.PortKey, &out.PortKey
+ *out = new(string)
+ **out = **in
+ }
}
// DeepCopy is an autogenerated deepcopy function, copying the receiver, creating a new ConnectionCredentialKey.
@@ -3109,6 +3048,41 @@ func (in *ProvisionStatements) DeepCopy() *ProvisionStatements {
return out
}
+// DeepCopyInto is an autogenerated deepcopy function, copying the receiver, writing into out. in must be non-nil.
+func (in *RSMSpec) DeepCopyInto(out *RSMSpec) {
+ *out = *in
+ if in.Roles != nil {
+ in, out := &in.Roles, &out.Roles
+ *out = make([]workloadsv1alpha1.ReplicaRole, len(*in))
+ copy(*out, *in)
+ }
+ if in.RoleProbe != nil {
+ in, out := &in.RoleProbe, &out.RoleProbe
+ *out = new(workloadsv1alpha1.RoleProbe)
+ (*in).DeepCopyInto(*out)
+ }
+ if in.MembershipReconfiguration != nil {
+ in, out := &in.MembershipReconfiguration, &out.MembershipReconfiguration
+ *out = new(workloadsv1alpha1.MembershipReconfiguration)
+ (*in).DeepCopyInto(*out)
+ }
+ if in.MemberUpdateStrategy != nil {
+ in, out := &in.MemberUpdateStrategy, &out.MemberUpdateStrategy
+ *out = new(workloadsv1alpha1.MemberUpdateStrategy)
+ **out = **in
+ }
+}
+
+// DeepCopy is an autogenerated deepcopy function, copying the receiver, creating a new RSMSpec.
+func (in *RSMSpec) DeepCopy() *RSMSpec {
+ if in == nil {
+ return nil
+ }
+ out := new(RSMSpec)
+ in.DeepCopyInto(out)
+ return out
+}
+
// DeepCopyInto is an autogenerated deepcopy function, copying the receiver, writing into out. in must be non-nil.
func (in *Reconfigure) DeepCopyInto(out *Reconfigure) {
*out = *in
@@ -3295,65 +3269,15 @@ func (in *RestoreFromSpec) DeepCopy() *RestoreFromSpec {
}
// DeepCopyInto is an autogenerated deepcopy function, copying the receiver, writing into out. in must be non-nil.
-func (in *RetentionSpec) DeepCopyInto(out *RetentionSpec) {
- *out = *in
- if in.TTL != nil {
- in, out := &in.TTL, &out.TTL
- *out = new(string)
- **out = **in
- }
-}
-
-// DeepCopy is an autogenerated deepcopy function, copying the receiver, creating a new RetentionSpec.
-func (in *RetentionSpec) DeepCopy() *RetentionSpec {
- if in == nil {
- return nil
- }
- out := new(RetentionSpec)
- in.DeepCopyInto(out)
- return out
-}
-
-// DeepCopyInto is an autogenerated deepcopy function, copying the receiver, writing into out. in must be non-nil.
-func (in *Schedule) DeepCopyInto(out *Schedule) {
+func (in *SchedulePolicy) DeepCopyInto(out *SchedulePolicy) {
*out = *in
- if in.StartingDeadlineMinutes != nil {
- in, out := &in.StartingDeadlineMinutes, &out.StartingDeadlineMinutes
- *out = new(int64)
- **out = **in
- }
- if in.Snapshot != nil {
- in, out := &in.Snapshot, &out.Snapshot
- *out = new(SchedulePolicy)
- **out = **in
- }
- if in.Datafile != nil {
- in, out := &in.Datafile, &out.Datafile
- *out = new(SchedulePolicy)
- **out = **in
- }
- if in.Logfile != nil {
- in, out := &in.Logfile, &out.Logfile
- *out = new(SchedulePolicy)
+ if in.Enabled != nil {
+ in, out := &in.Enabled, &out.Enabled
+ *out = new(bool)
**out = **in
}
}
-// DeepCopy is an autogenerated deepcopy function, copying the receiver, creating a new Schedule.
-func (in *Schedule) DeepCopy() *Schedule {
- if in == nil {
- return nil
- }
- out := new(Schedule)
- in.DeepCopyInto(out)
- return out
-}
-
-// DeepCopyInto is an autogenerated deepcopy function, copying the receiver, writing into out. in must be non-nil.
-func (in *SchedulePolicy) DeepCopyInto(out *SchedulePolicy) {
- *out = *in
-}
-
// DeepCopy is an autogenerated deepcopy function, copying the receiver, creating a new SchedulePolicy.
func (in *SchedulePolicy) DeepCopy() *SchedulePolicy {
if in == nil {
@@ -3442,6 +3366,11 @@ func (in *ScriptSpec) DeepCopyInto(out *ScriptSpec) {
*out = new(ScriptFrom)
(*in).DeepCopyInto(*out)
}
+ if in.Selector != nil {
+ in, out := &in.Selector, &out.Selector
+ *out = new(metav1.LabelSelector)
+ (*in).DeepCopyInto(*out)
+ }
}
// DeepCopy is an autogenerated deepcopy function, copying the receiver, creating a new ScriptSpec.
@@ -3691,27 +3620,6 @@ func (in *ShellTrigger) DeepCopy() *ShellTrigger {
return out
}
-// DeepCopyInto is an autogenerated deepcopy function, copying the receiver, writing into out. in must be non-nil.
-func (in *SnapshotPolicy) DeepCopyInto(out *SnapshotPolicy) {
- *out = *in
- in.BasePolicy.DeepCopyInto(&out.BasePolicy)
- if in.Hooks != nil {
- in, out := &in.Hooks, &out.Hooks
- *out = new(BackupPolicyHook)
- (*in).DeepCopyInto(*out)
- }
-}
-
-// DeepCopy is an autogenerated deepcopy function, copying the receiver, creating a new SnapshotPolicy.
-func (in *SnapshotPolicy) DeepCopy() *SnapshotPolicy {
- if in == nil {
- return nil
- }
- out := new(SnapshotPolicy)
- in.DeepCopyInto(out)
- return out
-}
-
// DeepCopyInto is an autogenerated deepcopy function, copying the receiver, writing into out. in must be non-nil.
func (in *StatefulSetSpec) DeepCopyInto(out *StatefulSetSpec) {
*out = *in
diff --git a/apis/dataprotection/v1alpha1/actionset_types.go b/apis/dataprotection/v1alpha1/actionset_types.go
new file mode 100644
index 00000000000..8dd1dc9455d
--- /dev/null
+++ b/apis/dataprotection/v1alpha1/actionset_types.go
@@ -0,0 +1,249 @@
+/*
+Copyright (C) 2022-2023 ApeCloud Co., Ltd
+
+Licensed under the Apache License, Version 2.0 (the "License");
+you may not use this file except in compliance with the License.
+You may obtain a copy of the License at
+
+ http://www.apache.org/licenses/LICENSE-2.0
+
+Unless required by applicable law or agreed to in writing, software
+distributed under the License is distributed on an "AS IS" BASIS,
+WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+See the License for the specific language governing permissions and
+limitations under the License.
+*/
+
+package v1alpha1
+
+import (
+ corev1 "k8s.io/api/core/v1"
+ metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
+)
+
+// ActionSetSpec defines the desired state of ActionSet
+type ActionSetSpec struct {
+ // backupType specifies the backup type, supported values: Full, Continuous.
+ // Full means full backup.
+ // Incremental means back up data that have changed since the last backup (full or incremental).
+ // Differential means back up data that have changed since the last full backup.
+ // Continuous will back up the transaction log continuously, the PITR (Point in Time Recovery).
+ // can be performed based on the continuous backup and full backup.
+ // +kubebuilder:validation:Enum={Full,Incremental,Differential,Continuous}
+ // +kubebuilder:default=Full
+ // +kubebuilder:validation:Required
+ BackupType BackupType `json:"backupType"`
+
+ // List of environment variables to set in the container.
+ // +kubebuilder:pruning:PreserveUnknownFields
+ // +optional
+ Env []corev1.EnvVar `json:"env,omitempty" patchStrategy:"merge" patchMergeKey:"name"`
+
+ // List of sources to populate environment variables in the container.
+ // The keys defined within a source must be a C_IDENTIFIER. All invalid keys
+ // will be reported as an event when the container is starting. When a key exists in multiple
+ // sources, the value associated with the last source will take precedence.
+ // Values defined by an Env with a duplicate key will take precedence.
+ // Cannot be updated.
+ // +kubebuilder:pruning:PreserveUnknownFields
+ // +optional
+ EnvFrom []corev1.EnvFromSource `json:"envFrom,omitempty"`
+
+ // backup specifies the backup action.
+ // +optional
+ Backup *BackupActionSpec `json:"backup,omitempty"`
+
+ // restore specifies the restore action.
+ // +optional
+ Restore *RestoreActionSpec `json:"restore,omitempty"`
+}
+
+// ActionSetStatus defines the observed state of ActionSet
+type ActionSetStatus struct {
+ // phase - in list of [Available,Unavailable]
+ // +optional
+ Phase Phase `json:"phase,omitempty"`
+
+ // A human-readable message indicating details about why the ActionSet is in this phase.
+ // +optional
+ Message string `json:"message,omitempty"`
+
+ // generation number
+ // +optional
+ ObservedGeneration int64 `json:"observedGeneration,omitempty"`
+}
+
+// BackupType the backup type.
+// +enum
+// +kubebuilder:validation:Enum={Full,Incremental,Differential,Continuous}
+type BackupType string
+
+const (
+ BackupTypeFull BackupType = "Full"
+ BackupTypeIncremental BackupType = "Incremental"
+ BackupTypeDifferential BackupType = "Differential"
+ BackupTypeContinuous BackupType = "Continuous"
+)
+
+type BackupActionSpec struct {
+ // backupData specifies the backup data action.
+ // +kubebuilder:validation:Required
+ BackupData *BackupDataActionSpec `json:"backupData,omitempty"`
+
+ // preBackup specifies a hook that should be executed before the backup.
+ // +optional
+ PreBackup []ActionSpec `json:"preBackup,omitempty"`
+
+ // postBackup specifies a hook that should be executed after the backup.
+ // +optional
+ PostBackup []ActionSpec `json:"postBackup,omitempty"`
+}
+
+// BackupDataActionSpec defines how to back up data.
+type BackupDataActionSpec struct {
+ JobActionSpec `json:",inline"`
+
+ // syncProgress specifies whether to sync the backup progress and its interval seconds.
+ // +optional
+ SyncProgress *SyncProgress `json:"syncProgress,omitempty"`
+}
+
+type SyncProgress struct {
+ // enabled specifies whether to sync the backup progress. If enabled,
+ // a sidecar container will be created to sync the backup progress to the
+ // Backup CR status.
+ // +optional
+ Enabled *bool `json:"enabled,omitempty"`
+
+ // intervalSeconds specifies the interval seconds to sync the backup progress.
+ // +optional
+ // +kubebuilder:default=60
+ IntervalSeconds *int32 `json:"intervalSeconds,omitempty"`
+}
+
+// RestoreActionSpec defines how to restore data.
+type RestoreActionSpec struct {
+ // prepareData specifies the action to prepare data.
+ // +optional
+ PrepareData *JobActionSpec `json:"prepareData,omitempty"`
+
+ // postReady specifies the action to execute after the data is ready.
+ // +optional
+ PostReady []ActionSpec `json:"postReady,omitempty"`
+}
+
+// ActionSpec defines an action that should be executed. Only one of the fields may be set.
+type ActionSpec struct {
+ // exec specifies the action should be executed by the pod exec API in a container.
+ // +optional
+ Exec *ExecActionSpec `json:"exec,omitempty"`
+
+ // job specifies the action should be executed by a Kubernetes Job.
+ // +optional
+ Job *JobActionSpec `json:"job,omitempty"`
+}
+
+// ExecActionSpec is an action that uses the pod exec API to execute a command in a container
+// in a pod.
+type ExecActionSpec struct {
+ // container is the container in the pod where the command should be executed.
+ // If not specified, the pod's first container is used.
+ // +optional
+ Container string `json:"container,omitempty"`
+
+ // Command is the command and arguments to execute.
+ // +kubebuilder:validation:MinItems=1
+ Command []string `json:"command"`
+
+ // OnError specifies how should behave if it encounters an error executing this action.
+ // +optional
+ // +kubebuilder:default=Fail
+ OnError ActionErrorMode `json:"onError,omitempty"`
+
+ // Timeout defines the maximum amount of time should wait for the hook to complete before
+ // considering the execution a failure.
+ // +optional
+ Timeout metav1.Duration `json:"timeout,omitempty"`
+}
+
+// JobActionSpec is an action that creates a Kubernetes Job to execute a command.
+type JobActionSpec struct {
+ // image specifies the image of backup container.
+ // +kubebuilder:validation:Required
+ Image string `json:"image"`
+
+ // runOnTargetPodNode specifies whether to run the job workload on the
+ // target pod node. If backup container should mount the target pod's
+ // volume, this field should be set to true.
+ // +optional
+ // +kubebuilder:default=false
+ RunOnTargetPodNode *bool `json:"runOnTargetPodNode,omitempty"`
+
+ // command specifies the commands to back up the volume data.
+ // +kubebuilder:validation:Required
+ Command []string `json:"command"`
+
+ // OnError specifies how should behave if it encounters an error executing
+ // this action.
+ // +optional
+ // +kubebuilder:default=Fail
+ OnError ActionErrorMode `json:"onError,omitempty"`
+}
+
+// ActionErrorMode defines how should treat an error from an action.
+// +kubebuilder:validation:Enum=Continue;Fail
+type ActionErrorMode string
+
+const (
+ // ActionErrorModeContinue means that an error from an action is acceptable.
+ ActionErrorModeContinue ActionErrorMode = "Continue"
+
+ // ActionErrorModeFail means that an error from an action is problematic.
+ ActionErrorModeFail ActionErrorMode = "Fail"
+)
+
+// +genclient
+// +genclient:nonNamespaced
+// +k8s:openapi-gen=true
+// +kubebuilder:object:root=true
+// +kubebuilder:subresource:status
+// +kubebuilder:resource:categories={kubeblocks},scope=Cluster,shortName=as
+// +kubebuilder:printcolumn:name="BACKUP-TYPE",type="string",JSONPath=".spec.backupType"
+// +kubebuilder:printcolumn:name="STATUS",type="string",JSONPath=".status.phase"
+// +kubebuilder:printcolumn:name="AGE",type="date",JSONPath=".metadata.creationTimestamp"
+
+// ActionSet is the Schema for the actionsets API
+type ActionSet struct {
+ metav1.TypeMeta `json:",inline"`
+ metav1.ObjectMeta `json:"metadata,omitempty"`
+
+ Spec ActionSetSpec `json:"spec,omitempty"`
+ Status ActionSetStatus `json:"status,omitempty"`
+}
+
+// +kubebuilder:object:root=true
+
+// ActionSetList contains a list of ActionSet
+type ActionSetList struct {
+ metav1.TypeMeta `json:",inline"`
+ metav1.ListMeta `json:"metadata,omitempty"`
+ Items []ActionSet `json:"items"`
+}
+
+func init() {
+ SchemeBuilder.Register(&ActionSet{}, &ActionSetList{})
+}
+
+func (r *ActionSet) HasPrepareDataStage() bool {
+ if r == nil || r.Spec.Restore == nil {
+ return false
+ }
+ return r.Spec.Restore.PrepareData != nil
+}
+
+func (r *ActionSet) HasPostReadyStage() bool {
+ if r == nil || r.Spec.Restore == nil {
+ return false
+ }
+ return len(r.Spec.Restore.PostReady) > 0
+}
diff --git a/apis/dataprotection/v1alpha1/backup_types.go b/apis/dataprotection/v1alpha1/backup_types.go
index b2218617b26..1698c10d0f0 100644
--- a/apis/dataprotection/v1alpha1/backup_types.go
+++ b/apis/dataprotection/v1alpha1/backup_types.go
@@ -17,175 +17,278 @@ limitations under the License.
package v1alpha1
import (
- "fmt"
- "sort"
-
- "golang.org/x/exp/slices"
+ corev1 "k8s.io/api/core/v1"
metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
)
// BackupSpec defines the desired state of Backup.
type BackupSpec struct {
- // Which backupPolicy is applied to perform this backup
+ // Which backupPolicy is applied to perform this backup.
// +kubebuilder:validation:Required
// +kubebuilder:validation:Pattern:=`^[a-z0-9]([a-z0-9\.\-]*[a-z0-9])?$`
BackupPolicyName string `json:"backupPolicyName"`
- // Backup Type. datafile or logfile or snapshot. If not set, datafile is the default type.
- // +kubebuilder:default=datafile
- BackupType BackupType `json:"backupType"`
+ // backupMethod specifies the backup method name that is defined in backupPolicy.
+ // +kubebuilder:validation:Required
+ BackupMethod string `json:"backupMethod"`
+
+ // deletionPolicy determines whether the backup contents stored in backup repository
+ // should be deleted when the backup custom resource is deleted.
+ // Supported values are "Retain" and "Delete".
+ // "Retain" means that the backup content and its physical snapshot on backup repository are kept.
+ // "Delete" means that the backup content and its physical snapshot on backup repository are deleted.
+ // +kubebuilder:validation:Enum=Delete;Retain
+ // +kubebuilder:validation:Required
+ // +kubebuilder:default=Delete
+ DeletionPolicy BackupDeletionPolicy `json:"deletionPolicy,omitempty"`
+
+ // retentionPeriod determines a duration up to which the backup should be kept.
+ // controller will remove all backups that are older than the RetentionPeriod.
+ // For example, RetentionPeriod of `30d` will keep only the backups of last 30 days.
+ // Sample duration format:
+ // - years: 2y
+ // - months: 6mo
+ // - days: 30d
+ // - hours: 12h
+ // - minutes: 30m
+ // You can also combine the above durations. For example: 30d12h30m
+ // +kubebuilder:default="7d"
+ // +optional
+ RetentionPeriod RetentionPeriod `json:"retentionPeriod,omitempty"`
- // if backupType is incremental, parentBackupName is required.
+ // parentBackupName determines the parent backup name for incremental or
+ // differential backup.
// +optional
ParentBackupName string `json:"parentBackupName,omitempty"`
}
// BackupStatus defines the observed state of Backup.
type BackupStatus struct {
+ // formatVersion is the backup format version, including major, minor and patch version.
// +optional
- Phase BackupPhase `json:"phase,omitempty"`
+ FormatVersion string `json:"formatVersion,omitempty"`
- // Records parentBackupName if backupType is incremental.
+ // phase is the current state of the Backup.
// +optional
- ParentBackupName string `json:"parentBackupName,omitempty"`
+ Phase BackupPhase `json:"phase,omitempty"`
- // The date and time when the Backup is eligible for garbage collection.
- // 'null' means the Backup is NOT be cleaned except delete manual.
+ // expiration is when this backup is eligible for garbage collection.
+ // 'null' means the Backup will NOT be cleaned except delete manual.
// +optional
Expiration *metav1.Time `json:"expiration,omitempty"`
- // Date/time when the backup started being processed.
+ // startTimestamp records the time a backup was started.
+ // The server's time is used for StartTimestamp.
// +optional
StartTimestamp *metav1.Time `json:"startTimestamp,omitempty"`
- // Date/time when the backup finished being processed.
+ // completionTimestamp records the time a backup was completed.
+ // Completion time is recorded even on failed backups.
+ // The server's time is used for CompletionTimestamp.
// +optional
CompletionTimestamp *metav1.Time `json:"completionTimestamp,omitempty"`
// The duration time of backup execution.
- // When converted to a string, the form is "1h2m0.5s".
+ // When converted to a string, the format is "1h2m0.5s".
// +optional
Duration *metav1.Duration `json:"duration,omitempty"`
- // Backup total size.
- // A string with capacity units in the form of "1Gi", "1Mi", "1Ki".
+ // totalSize is the total size of backed up data size.
+ // A string with capacity units in the format of "1Gi", "1Mi", "1Ki".
// +optional
TotalSize string `json:"totalSize,omitempty"`
- // The reason for a backup failure.
+ // failureReason is an error that caused the backup to fail.
// +optional
FailureReason string `json:"failureReason,omitempty"`
- // remoteVolume saves the backup data.
+ // backupRepoName is the name of the backup repository.
+ // +optional
+ BackupRepoName string `json:"backupRepoName,omitempty"`
+
+ // path is the directory inside the backup repository where the backup data is stored.
+ // It is an absolute path in the backup repository.
+ // +optional
+ Path string `json:"path,omitempty"`
+
+ // persistentVolumeClaimName is the name of the persistent volume claim that
+ // is used to store the backup data.
// +optional
PersistentVolumeClaimName string `json:"persistentVolumeClaimName,omitempty"`
- // logFilePersistentVolumeClaimName saves the logfile backup data.
+ // timeRange records the time range of backed up data, for PITR, this is the
+ // time range of recoverable data.
// +optional
- LogFilePersistentVolumeClaimName string `json:"logFilePersistentVolumeClaimName,omitempty"`
+ TimeRange *BackupTimeRange `json:"timeRange,omitempty"`
- // backupToolName references the backup tool name.
+ // target records the target information for this backup.
// +optional
- BackupToolName string `json:"backupToolName,omitempty"`
+ Target *BackupTarget `json:"target,omitempty"`
- // sourceCluster records the source cluster information for this backup.
- SourceCluster string `json:"sourceCluster,omitempty"`
+ // backupMethod records the backup method information for this backup.
+ // Refer to BackupMethod for more details.
+ // +optional
+ BackupMethod *BackupMethod `json:"backupMethod,omitempty"`
- // availableReplicas available replicas for statefulSet which created by backup.
+ // actions records the actions information for this backup.
// +optional
- AvailableReplicas *int32 `json:"availableReplicas,omitempty"`
+ Actions []ActionStatus `json:"actions,omitempty"`
- // manifests determines the backup metadata info.
+ // volumeSnapshots records the volume snapshot status for the action.
// +optional
- Manifests *ManifestsStatus `json:"manifests,omitempty"`
+ VolumeSnapshots []VolumeSnapshotStatus `json:"volumeSnapshots,omitempty"`
}
-type ManifestsStatus struct {
- // backupLog records startTime and stopTime of data logging.
+// BackupTimeRange records the time range of backed up data, for PITR, this is the
+// time range of recoverable data.
+type BackupTimeRange struct {
+ // start records the start time of backup.
// +optional
- BackupLog *BackupLogStatus `json:"backupLog,omitempty"`
+ Start *metav1.Time `json:"start,omitempty"`
- // target records the target cluster metadata string, which is in JSON format.
+ // end records the end time of backup.
// +optional
- Target string `json:"target,omitempty"`
+ End *metav1.Time `json:"end,omitempty"`
+}
- // snapshot records the volume snapshot metadata.
- // +optional
- Snapshot *BackupSnapshotStatus `json:"backupSnapshot,omitempty"`
+// BackupDeletionPolicy describes a policy for end-of-life maintenance of backup content.
+// +enum
+// +kubebuilder:validation:Enum={Delete,Retain}
+type BackupDeletionPolicy string
+
+const (
+ BackupDeletionPolicyDelete BackupDeletionPolicy = "Delete"
+ BackupDeletionPolicyRetain BackupDeletionPolicy = "Retain"
+)
+
+// BackupPhase is a string representation of the lifecycle phase of a Backup.
+// +enum
+// +kubebuilder:validation:Enum={New,InProgress,Running,Completed,Failed,Deleting}
+type BackupPhase string
+
+const (
+ // BackupPhaseNew means the backup has been created but not yet processed by
+ // the BackupController.
+ BackupPhaseNew BackupPhase = "New"
+
+ // BackupPhaseRunning means the backup is currently executing.
+ BackupPhaseRunning BackupPhase = "Running"
+
+ // BackupPhaseCompleted means the backup has run successfully without errors.
+ BackupPhaseCompleted BackupPhase = "Completed"
- // backupTool records information about backup files generated by the backup tool.
+ // BackupPhaseFailed means the backup ran but encountered an error that
+ // prevented it from completing successfully.
+ BackupPhaseFailed BackupPhase = "Failed"
+
+ // BackupPhaseDeleting means the backup and all its associated data are being deleted.
+ BackupPhaseDeleting BackupPhase = "Deleting"
+)
+
+type ActionStatus struct {
+ // name is the name of the action.
+ Name string `json:"name,omitempty"`
+
+ // phase is the current state of the action.
// +optional
- BackupTool *BackupToolManifestsStatus `json:"backupTool,omitempty"`
+ Phase ActionPhase `json:"phase,omitempty"`
- // userContext stores some loosely structured and extensible information.
+ // startTimestamp records the time an action was started.
// +optional
- UserContext map[string]string `json:"userContext,omitempty"`
-}
+ StartTimestamp *metav1.Time `json:"startTimestamp,omitempty"`
-type BackupLogStatus struct {
- // startTime records the start time of data logging.
+ // completionTimestamp records the time an action was completed.
// +optional
- StartTime *metav1.Time `json:"startTime,omitempty"`
+ CompletionTimestamp *metav1.Time `json:"completionTimestamp,omitempty"`
- // stopTime records the stop time of data logging.
+ // failureReason is an error that caused the backup to fail.
// +optional
- StopTime *metav1.Time `json:"stopTime,omitempty"`
-}
+ FailureReason string `json:"failureReason,omitempty"`
-type BackupSnapshotStatus struct {
- // volumeSnapshotName records the volumeSnapshot name.
+ // actionType is the type of the action.
// +optional
- VolumeSnapshotName string `json:"volumeSnapshotName,omitempty"`
+ ActionType ActionType `json:"actionType,omitempty"`
- // volumeSnapshotContentName specifies the name of a pre-existing VolumeSnapshotContent
- // object representing an existing volume snapshot.
- // This field should be set if the snapshot already exists and only needs a representation in Kubernetes.
- // This field is immutable.
+ // availableReplicas available replicas for statefulSet action.
// +optional
- VolumeSnapshotContentName string `json:"volumeSnapshotContentName,omitempty"`
-}
+ AvailableReplicas *int32 `json:"availableReplicas,omitempty"`
-type BackupToolManifestsStatus struct {
- // filePath records the file path of backup.
+ // objectRef is the object reference for the action.
// +optional
- FilePath string `json:"filePath,omitempty"`
+ ObjectRef *corev1.ObjectReference `json:"objectRef,omitempty"`
- // logFilePath records the log file path of backup.
+ // totalSize is the total size of backed up data size.
+ // A string with capacity units in the format of "1Gi", "1Mi", "1Ki".
// +optional
- LogFilePath string `json:"logFilePath,omitempty"`
+ TotalSize string `json:"totalSize,omitempty"`
- // volumeName records volume name of backup data pvc.
+ // timeRange records the time range of backed up data, for PITR, this is the
+ // time range of recoverable data.
// +optional
- VolumeName string `json:"volumeName,omitempty"`
+ TimeRange *BackupTimeRange `json:"timeRange,omitempty"`
- // Backup upload total size.
- // A string with capacity units in the form of "1Gi", "1Mi", "1Ki".
+ // volumeSnapshots records the volume snapshot status for the action.
// +optional
- UploadTotalSize string `json:"uploadTotalSize,omitempty"`
+ VolumeSnapshots []VolumeSnapshotStatus `json:"volumeSnapshots,omitempty"`
+}
- // checksum of backup file, generated by md5 or sha1 or sha256.
+type VolumeSnapshotStatus struct {
+ // name is the name of the volume snapshot.
+ Name string `json:"name,omitempty"`
+
+ // contentName is the name of the volume snapshot content.
+ ContentName string `json:"contentName,omitempty"`
+
+ // volumeName is the name of the volume.
// +optional
- Checksum string `json:"checksum,omitempty"`
+ VolumeName string `json:"volumeName,omitempty"`
- // backup checkpoint, for incremental backup.
+ // size is the size of the volume snapshot.
// +optional
- Checkpoint string `json:"checkpoint,omitempty"`
+ Size string `json:"size,omitempty"`
}
+type ActionPhase string
+
+const (
+ // ActionPhaseNew means the action has been created but not yet processed by
+ // the BackupController.
+ ActionPhaseNew ActionPhase = "New"
+
+ // ActionPhaseRunning means the action is currently executing.
+ ActionPhaseRunning ActionPhase = "Running"
+
+ // ActionPhaseCompleted means the action has run successfully without errors.
+ ActionPhaseCompleted ActionPhase = "Completed"
+
+ // ActionPhaseFailed means the action ran but encountered an error that
+ ActionPhaseFailed ActionPhase = "Failed"
+)
+
+type ActionType string
+
+const (
+ ActionTypeJob ActionType = "Job"
+ ActionTypeStatefulSet ActionType = "StatefulSet"
+ ActionTypeNone ActionType = ""
+)
+
// +genclient
// +k8s:openapi-gen=true
// +kubebuilder:object:root=true
// +kubebuilder:subresource:status
// +kubebuilder:resource:categories={kubeblocks},scope=Namespaced
-// +kubebuilder:printcolumn:name="TYPE",type=string,JSONPath=`.spec.backupType`
+// +kubebuilder:printcolumn:name="POLICY",type=string,JSONPath=`.spec.backupPolicyName`
+// +kubebuilder:printcolumn:name="METHOD",type=string,JSONPath=`.spec.backupMethod`
+// +kubebuilder:printcolumn:name="REPO",type=string,JSONPath=`.status.backupRepoName`
// +kubebuilder:printcolumn:name="STATUS",type=string,JSONPath=`.status.phase`
-// +kubebuilder:printcolumn:name="SOURCE-CLUSTER",type=string,JSONPath=`.status.sourceCluster`
// +kubebuilder:printcolumn:name="TOTAL-SIZE",type=string,JSONPath=`.status.totalSize`
// +kubebuilder:printcolumn:name="DURATION",type=string,JSONPath=`.status.duration`
-// +kubebuilder:printcolumn:name="CREATE-TIME",type=string,JSONPath=".metadata.creationTimestamp"
+// +kubebuilder:printcolumn:name="CREATION-TIME",type=string,JSONPath=".metadata.creationTimestamp"
// +kubebuilder:printcolumn:name="COMPLETION-TIME",type=string,JSONPath=`.status.completionTimestamp`
+// +kubebuilder:printcolumn:name="EXPIRATION-TIME",type=string,JSONPath=`.status.expiration`
-// Backup is the Schema for the backups API (defined by User).
+// Backup is the Schema for the backups API.
type Backup struct {
metav1.TypeMeta `json:",inline"`
metav1.ObjectMeta `json:"metadata,omitempty"`
@@ -207,96 +310,22 @@ func init() {
SchemeBuilder.Register(&Backup{}, &BackupList{})
}
-// Validate validates the BackupSpec and returns an error if invalid.
-func (r *BackupSpec) Validate(backupPolicy *BackupPolicy) error {
- notSupportedMessage := "backupPolicy: %s not supports %s backup in backupPolicy"
- switch r.BackupType {
- case BackupTypeSnapshot:
- if backupPolicy.Spec.Snapshot == nil {
- return fmt.Errorf(notSupportedMessage, r.BackupPolicyName, BackupTypeSnapshot)
- }
- case BackupTypeDataFile:
- if backupPolicy.Spec.Datafile == nil {
- return fmt.Errorf(notSupportedMessage, r.BackupPolicyName, BackupTypeDataFile)
- }
- case BackupTypeLogFile:
- if backupPolicy.Spec.Logfile == nil {
- return fmt.Errorf(notSupportedMessage, r.BackupPolicyName, BackupTypeLogFile)
- }
- }
- return nil
-}
-
-// GetStartTime gets the backup start time. the default return is status.startTime, unless status.manifests.backupLog.startTime is not nil.
-func (r *BackupStatus) GetStartTime() *metav1.Time {
- if r.Manifests != nil && r.Manifests.BackupLog != nil && r.Manifests.BackupLog.StartTime != nil {
- return r.Manifests.BackupLog.StartTime
- }
- return r.StartTimestamp
-}
-
-// GetStopTime gets the backup stop time. the default return is status.completionTimestamp, unless status.manifests.backupLog.stopTime is not nil.
-func (r *BackupStatus) GetStopTime() *metav1.Time {
- if r.Manifests != nil && r.Manifests.BackupLog != nil && r.Manifests.BackupLog.StopTime != nil {
- return r.Manifests.BackupLog.StopTime
+// GetStartTime gets the backup start time. Default return status.startTimestamp,
+// unless status.timeRange.startTime is not nil.
+func (r *Backup) GetStartTime() *metav1.Time {
+ s := r.Status
+ if s.TimeRange != nil && s.TimeRange.Start != nil {
+ return s.TimeRange.Start
}
- return r.CompletionTimestamp
+ return s.StartTimestamp
}
-// GetRecoverableTimeRange returns the recoverable time range array.
-func GetRecoverableTimeRange(backups []Backup) []BackupLogStatus {
- sort.Slice(backups, func(i, j int) bool {
- if backups[i].Status.StartTimestamp == nil && backups[j].Status.StartTimestamp != nil {
- return false
- }
- if backups[i].Status.StartTimestamp != nil && backups[j].Status.StartTimestamp == nil {
- return true
- }
- if backups[i].Status.StartTimestamp.Equal(backups[j].Status.StartTimestamp) {
- return backups[i].Name < backups[j].Name
- }
- return backups[i].Status.StartTimestamp.Before(backups[j].Status.StartTimestamp)
- })
- getLogfileStartTimeAndStopTime := func() (*metav1.Time, *metav1.Time) {
- var (
- startTime *metav1.Time
- stopTime *metav1.Time
- )
- for _, b := range backups {
- if b.Spec.BackupType != BackupTypeLogFile {
- continue
- }
- startTime = b.Status.GetStartTime()
- stopTime = b.Status.GetStopTime()
- break
- }
- return startTime, stopTime
- }
- logfileStartTime, logfileStopTime := getLogfileStartTimeAndStopTime()
- // if not exists the startTime/stopTime of the first log file, return
- if logfileStartTime.IsZero() || logfileStopTime.IsZero() {
- return nil
- }
- getFirstRecoverableBaseBackup := func() *Backup {
- for _, b := range backups {
- if !slices.Contains([]BackupType{BackupTypeDataFile, BackupTypeSnapshot}, b.Spec.BackupType) ||
- b.Status.Phase != BackupCompleted {
- continue
- }
- backupStopTime := b.Status.GetStopTime()
- // checks if the baseBackup stop time is between logfileStartTime and logfileStopTime.
- if !backupStopTime.Before(logfileStartTime) &&
- backupStopTime.Before(logfileStopTime) {
- return &b
- }
- }
- return nil
- }
- firstRecoverableBaseBackup := getFirstRecoverableBaseBackup()
- if firstRecoverableBaseBackup == nil {
- return nil
+// GetEndTime gets the backup end time. Default return status.completionTimestamp,
+// unless status.timeRange.endTime is not nil.
+func (r *Backup) GetEndTime() *metav1.Time {
+ s := r.Status
+ if s.TimeRange != nil && s.TimeRange.End != nil {
+ return s.TimeRange.End
}
- // range of recoverable time
- return []BackupLogStatus{{StopTime: logfileStopTime,
- StartTime: firstRecoverableBaseBackup.Status.GetStopTime()}}
+ return s.CompletionTimestamp
}
diff --git a/apis/dataprotection/v1alpha1/backup_types_test.go b/apis/dataprotection/v1alpha1/backup_types_test.go
deleted file mode 100644
index c5db0709711..00000000000
--- a/apis/dataprotection/v1alpha1/backup_types_test.go
+++ /dev/null
@@ -1,170 +0,0 @@
-/*
-Copyright (C) 2022-2023 ApeCloud Co., Ltd
-
-Licensed under the Apache License, Version 2.0 (the "License");
-you may not use this file except in compliance with the License.
-You may obtain a copy of the License at
-
- http://www.apache.org/licenses/LICENSE-2.0
-
-Unless required by applicable law or agreed to in writing, software
-distributed under the License is distributed on an "AS IS" BASIS,
-WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
-See the License for the specific language governing permissions and
-limitations under the License.
-*/
-
-package v1alpha1
-
-import (
- "testing"
- "time"
-
- . "github.com/onsi/gomega"
-
- metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
-)
-
-func TestValidate(t *testing.T) {
- g := NewGomegaWithT(t)
-
- backupPolicy := &BackupPolicy{Spec: BackupPolicySpec{Snapshot: &SnapshotPolicy{}}}
- backupSpec := &BackupSpec{BackupType: BackupTypeSnapshot}
- g.Expect(backupSpec.Validate(backupPolicy)).Should(Succeed())
-
- backupPolicy = &BackupPolicy{}
- backupSpec = &BackupSpec{BackupType: BackupTypeSnapshot}
- g.Expect(backupSpec.Validate(backupPolicy)).Should(HaveOccurred())
-
- backupSpec = &BackupSpec{BackupType: BackupTypeDataFile}
- g.Expect(backupSpec.Validate(backupPolicy)).Should(HaveOccurred())
-
- backupSpec = &BackupSpec{BackupType: BackupTypeLogFile}
- g.Expect(backupSpec.Validate(backupPolicy)).Should(HaveOccurred())
-}
-
-func TestGetRecoverableTimeRange(t *testing.T) {
- g := NewGomegaWithT(t)
-
- // test empty backups
- emptyBackups := []Backup{}
- g.Expect(GetRecoverableTimeRange(emptyBackups)).Should(BeEmpty())
-
- now := metav1.Now()
- backupSnapshot := Backup{
- ObjectMeta: metav1.ObjectMeta{Name: "backup-snapshot"},
- Spec: BackupSpec{BackupType: BackupTypeSnapshot},
- Status: BackupStatus{
- Phase: BackupCompleted,
- Manifests: &ManifestsStatus{
- BackupLog: &BackupLogStatus{
- StartTime: &now,
- StopTime: &now,
- },
- },
- },
- }
- noStartTimeSnapshotBackup := Backup{
- ObjectMeta: metav1.ObjectMeta{Name: "backup-snapshot1"},
- Spec: BackupSpec{BackupType: BackupTypeSnapshot},
- Status: BackupStatus{
- Phase: BackupCompleted,
- Manifests: &ManifestsStatus{
- BackupLog: &BackupLogStatus{
- StopTime: &now,
- },
- },
- },
- }
- noTimeSnapshotBackup := Backup{
- ObjectMeta: metav1.ObjectMeta{Name: "backup-snapshot2"},
- Spec: BackupSpec{BackupType: BackupTypeSnapshot},
- Status: BackupStatus{
- Phase: BackupCompleted,
- },
- }
- stopTimeGTLogFileStopTimeBaseBackup := Backup{
- ObjectMeta: metav1.ObjectMeta{Name: "backup-snapshot2"},
- Spec: BackupSpec{BackupType: BackupTypeSnapshot},
- Status: BackupStatus{
- Phase: BackupCompleted,
- Manifests: &ManifestsStatus{
- BackupLog: &BackupLogStatus{
- StartTime: &now,
- StopTime: &metav1.Time{Time: now.Add(time.Minute * 61)},
- },
- },
- },
- }
- backupLogfile := Backup{
- ObjectMeta: metav1.ObjectMeta{Name: "backup-logfile"},
- Spec: BackupSpec{BackupType: BackupTypeLogFile},
- Status: BackupStatus{
- Phase: BackupCompleted,
- Manifests: &ManifestsStatus{
- BackupLog: &BackupLogStatus{
- StartTime: &now,
- StopTime: &metav1.Time{Time: now.Add(time.Hour)},
- },
- },
- },
- }
-
- backups := []Backup{backupSnapshot, backupLogfile, {}}
- g.Expect(GetRecoverableTimeRange(backups)).ShouldNot(BeEmpty())
-
- backups = []Backup{noStartTimeSnapshotBackup, backupLogfile, {}}
- g.Expect(GetRecoverableTimeRange(backups)).ShouldNot(BeEmpty())
-
- backups = []Backup{backupLogfile, noTimeSnapshotBackup}
- g.Expect(GetRecoverableTimeRange(backups)).Should(BeEmpty())
-
- backups = []Backup{backupLogfile, stopTimeGTLogFileStopTimeBaseBackup}
- g.Expect(GetRecoverableTimeRange(backups)).Should(BeEmpty())
-}
-
-func TestGetStartTime(t *testing.T) {
- startTimestamp := metav1.Now()
- backTimeRangeStartTime := metav1.Time{Time: time.Now().Add(10 * time.Second)}
- backup := Backup{
- ObjectMeta: metav1.ObjectMeta{Name: "backup1"},
- Spec: BackupSpec{BackupType: BackupTypeSnapshot},
- Status: BackupStatus{
- Phase: BackupCompleted,
- StartTimestamp: &startTimestamp,
- Manifests: &ManifestsStatus{
- BackupLog: &BackupLogStatus{
- StartTime: &backTimeRangeStartTime,
- },
- },
- },
- }
- g := NewGomegaWithT(t)
- g.Expect(backup.Status.GetStartTime().Second()).Should(Equal(backTimeRangeStartTime.Second()))
-
- backup.Status.Manifests.BackupLog.StartTime = nil
- g.Expect(backup.Status.GetStartTime().Second()).Should(Equal(startTimestamp.Second()))
-}
-
-func TestGetStopTime(t *testing.T) {
- stopTimestamp := metav1.Now()
- backTimeRangeStopTime := metav1.Time{Time: time.Now().Add(10 * time.Second)}
- backup := Backup{
- ObjectMeta: metav1.ObjectMeta{Name: "backup1"},
- Spec: BackupSpec{BackupType: BackupTypeSnapshot},
- Status: BackupStatus{
- Phase: BackupCompleted,
- CompletionTimestamp: &stopTimestamp,
- Manifests: &ManifestsStatus{
- BackupLog: &BackupLogStatus{
- StopTime: &backTimeRangeStopTime,
- },
- },
- },
- }
- g := NewGomegaWithT(t)
- g.Expect(backup.Status.GetStopTime().Second()).Should(Equal(backTimeRangeStopTime.Second()))
-
- backup.Status.Manifests.BackupLog.StopTime = nil
- g.Expect(backup.Status.GetStopTime().Second()).Should(Equal(stopTimestamp.Second()))
-}
diff --git a/apis/dataprotection/v1alpha1/backuppolicy_types.go b/apis/dataprotection/v1alpha1/backuppolicy_types.go
index 727cb72d70b..89ecc1a7160 100644
--- a/apis/dataprotection/v1alpha1/backuppolicy_types.go
+++ b/apis/dataprotection/v1alpha1/backuppolicy_types.go
@@ -17,292 +17,222 @@ limitations under the License.
package v1alpha1
import (
- "fmt"
- "strconv"
- "strings"
- "time"
-
- "k8s.io/apimachinery/pkg/api/resource"
+ corev1 "k8s.io/api/core/v1"
metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
)
// BackupPolicySpec defines the desired state of BackupPolicy
type BackupPolicySpec struct {
- // retention describe how long the Backup should be retained. if not set, will be retained forever.
- // +optional
- Retention *RetentionSpec `json:"retention,omitempty"`
-
- // schedule policy for backup.
- // +optional
- Schedule Schedule `json:"schedule,omitempty"`
-
- // the policy for snapshot backup.
- // +optional
- Snapshot *SnapshotPolicy `json:"snapshot,omitempty"`
-
- // the policy for datafile backup.
- // +optional
- Datafile *CommonBackupPolicy `json:"datafile,omitempty"`
-
- // the policy for logfile backup.
+ // backupRepoName is the name of BackupRepo and the backup data will be
+ // stored in this repository. If not set, will be stored in the default
+ // backup repository.
+ // +kubebuilder:validation:Pattern:=`^[a-z0-9]([a-z0-9\.\-]*[a-z0-9])?$`
// +optional
- Logfile *CommonBackupPolicy `json:"logfile,omitempty"`
-}
+ BackupRepoName *string `json:"backupRepoName,omitempty"`
-type RetentionSpec struct {
- // ttl is a time string ending with the 'd'|'D'|'h'|'H' character to describe how long
- // the Backup should be retained. if not set, will be retained forever.
- // +kubebuilder:validation:Pattern:=`^\d+[d|D|h|H]$`
+ // pathPrefix is the directory inside the backup repository to store the backup content.
+ // It is a relative to the path of the backup repository.
// +optional
- TTL *string `json:"ttl,omitempty"`
-}
+ PathPrefix string `json:"pathPrefix,omitempty"`
-type Schedule struct {
- // startingDeadlineMinutes defines the deadline in minutes for starting the backup job
- // if it misses scheduled time for any reason.
+ // Specifies the number of retries before marking the backup failed.
// +optional
// +kubebuilder:validation:Minimum=0
- // +kubebuilder:validation:Maximum=1440
- StartingDeadlineMinutes *int64 `json:"startingDeadlineMinutes,omitempty"`
-
- // schedule policy for snapshot backup.
- // +optional
- Snapshot *SchedulePolicy `json:"snapshot,omitempty"`
-
- // schedule policy for datafile backup.
- // +optional
- Datafile *SchedulePolicy `json:"datafile,omitempty"`
-
- // schedule policy for logfile backup.
- // +optional
- Logfile *SchedulePolicy `json:"logfile,omitempty"`
-}
+ // +kubebuilder:validation:Maximum=10
+ BackoffLimit *int32 `json:"backoffLimit,omitempty"`
-type SchedulePolicy struct {
- // the cron expression for schedule, the timezone is in UTC. see https://en.wikipedia.org/wiki/Cron.
+ // target specifies the target information to back up.
// +kubebuilder:validation:Required
- CronExpression string `json:"cronExpression"`
+ Target *BackupTarget `json:"target"`
- // enable or disable the schedule.
+ // backupMethods defines the backup methods.
// +kubebuilder:validation:Required
- Enable bool `json:"enable"`
+ BackupMethods []BackupMethod `json:"backupMethods"`
}
-type SnapshotPolicy struct {
- BasePolicy `json:",inline"`
+type BackupTarget struct {
+ // podSelector is used to find the target pod. The volumes of the target pod
+ // will be backed up.
+ // +kube:validation:Required
+ PodSelector *PodSelector `json:"podSelector,omitempty"`
- // execute hook commands for backup.
+ // connectionCredential specifies the connection credential to connect to the
+ // target database cluster.
// +optional
- Hooks *BackupPolicyHook `json:"hooks,omitempty"`
-}
+ ConnectionCredential *ConnectionCredential `json:"connectionCredential,omitempty"`
-type CommonBackupPolicy struct {
- BasePolicy `json:",inline"`
-
- // refer to PersistentVolumeClaim and the backup data will be stored in the corresponding persistent volume.
+ // resources specifies the kubernetes resources to back up.
// +optional
- PersistentVolumeClaim PersistentVolumeClaim `json:"persistentVolumeClaim,omitempty"`
+ Resources *KubeResources `json:"resources,omitempty"`
- // refer to BackupRepo and the backup data will be stored in the corresponding repo.
- // +kubebuilder:validation:Pattern:=`^[a-z0-9]([a-z0-9\.\-]*[a-z0-9])?$`
- // +optional
- BackupRepoName *string `json:"backupRepoName,omitempty"`
-
- // which backup tool to perform database backup, only support one tool.
+ // serviceAccountName specifies the service account to run the backup workload.
// +kubebuilder:validation:Required
- // +kubebuilder:validation:Pattern:=`^[a-z0-9]([a-z0-9\.\-]*[a-z0-9])?$`
- BackupToolName string `json:"backupToolName,omitempty"`
+ ServiceAccountName string `json:"serviceAccountName,omitempty"`
}
-type PersistentVolumeClaim struct {
- // the name of PersistentVolumeClaim to store backup data.
- // +optional
- Name *string `json:"name,omitempty"`
+type PodSelector struct {
+ // labelsSelector is the label selector to filter the target pods.
+ *metav1.LabelSelector `json:",inline"`
- // storageClassName is the name of the StorageClass required by the claim.
- // +kubebuilder:validation:Pattern:=`^[a-z0-9]([a-z0-9\.\-]*[a-z0-9])?$`
- // +optional
- StorageClassName *string `json:"storageClassName,omitempty"`
+ // strategy specifies the strategy to select the target pod when multiple pods
+ // are selected.
+ // Valid values are:
+ // - All: select all pods that match the labelsSelector.
+ // - Any: select any one pod that match the labelsSelector.
+ // +kubebuilder:default=Any
+ Strategy PodSelectionStrategy `json:"strategy,omitempty"`
+}
- // initCapacity represents the init storage size of the PersistentVolumeClaim which should be created if not exist.
- // and the default value is 100Gi if it is empty.
- // +optional
- InitCapacity resource.Quantity `json:"initCapacity,omitempty"`
+// PodSelectionStrategy specifies the strategy to select when multiple pods are
+// selected for backup target
+// +kubebuilder:validation:Enum=All;Any
+type PodSelectionStrategy string
- // createPolicy defines the policy for creating the PersistentVolumeClaim, enum values:
- // - Never: do nothing if the PersistentVolumeClaim not exists.
- // - IfNotPresent: create the PersistentVolumeClaim if not present and the accessModes only contains 'ReadWriteMany'.
- // +kubebuilder:default=IfNotPresent
- // +optional
- CreatePolicy CreatePVCPolicy `json:"createPolicy,omitempty"`
-
- // persistentVolumeConfigMap references the configmap which contains a persistentVolume template.
- // key must be "persistentVolume" and value is the "PersistentVolume" struct.
- // support the following built-in Objects:
- // - $(GENERATE_NAME): generate a specific format "`PVC NAME`-`PVC NAMESPACE`".
- // if the PersistentVolumeClaim not exists and CreatePolicy is "IfNotPresent", the controller
- // will create it by this template. this is a mutually exclusive setting with "storageClassName".
- // +optional
- PersistentVolumeConfigMap *PersistentVolumeConfigMap `json:"persistentVolumeConfigMap,omitempty"`
-}
+const (
+ // PodSelectionStrategyAll selects all pods that match the labelsSelector.
+ PodSelectionStrategyAll PodSelectionStrategy = "All"
-type PersistentVolumeConfigMap struct {
- // the name of the persistentVolume ConfigMap.
- // +kubebuilder:validation:Required
- // +kubebuilder:validation:Pattern:=`^[a-z0-9]([a-z0-9\.\-]*[a-z0-9])?$`
- Name string `json:"name"`
+ // PodSelectionStrategyAny selects any one pod that match the labelsSelector.
+ PodSelectionStrategyAny PodSelectionStrategy = "Any"
+)
- // the namespace of the persistentVolume ConfigMap.
- // +kubebuilder:validation:Required
- // +kubebuilder:validation:Pattern:=`^[a-z0-9]([a-z0-9\-]*[a-z0-9])?$`
- Namespace string `json:"namespace"`
-}
+// ConnectionCredential specifies the connection credential to connect to the
+// target database cluster.
+type ConnectionCredential struct {
+ // secretName refers to the Secret object that contains the connection credential.
+ // +kube:validation:Required
+ SecretName string `json:"secretName,omitempty"`
-type BasePolicy struct {
- // target database cluster for backup.
- // +kubebuilder:validation:Required
- Target TargetCluster `json:"target"`
+ // usernameKey specifies the map key of the user in the connection credential secret.
+ // +kubebuilder:default=username
+ UsernameKey string `json:"usernameKey,omitempty"`
- // the number of automatic backups to retain. Value must be non-negative integer.
- // 0 means NO limit on the number of backups.
- // +kubebuilder:default=7
- // +optional
- BackupsHistoryLimit int32 `json:"backupsHistoryLimit,omitempty"`
+ // passwordKey specifies the map key of the password in the connection credential secret.
+ // +kubebuilder:default=password
+ PasswordKey string `json:"passwordKey,omitempty"`
- // count of backup stop retries on fail.
- // +optional
- OnFailAttempted int32 `json:"onFailAttempted,omitempty"`
+ // hostKey specifies the map key of the host in the connection credential secret.
+ HostKey string `json:"hostKey,omitempty"`
- // define how to update metadata for backup status.
- // +optional
- BackupStatusUpdates []BackupStatusUpdate `json:"backupStatusUpdates,omitempty"`
+ // portKey specifies the map key of the port in the connection credential secret.
+ // +kubebuilder:default=port
+ PortKey string `json:"portKey,omitempty"`
}
-// TargetCluster TODO (dsj): target cluster need redefined from Cluster API
-type TargetCluster struct {
- // labelsSelector is used to find matching pods.
- // Pods that match this label selector are counted to determine the number of pods
- // in their corresponding topology domain.
- // +kubebuilder:validation:Required
- // +kubebuilder:pruning:PreserveUnknownFields
- LabelsSelector *metav1.LabelSelector `json:"labelsSelector"`
+// KubeResources defines the kubernetes resources to back up.
+type KubeResources struct {
+ // selector is a metav1.LabelSelector to filter the target kubernetes resources
+ // that need to be backed up.
+ // If not set, will do not back up any kubernetes resources.
+ // +kube:validation:Required
+ Selector *metav1.LabelSelector `json:"selector,omitempty"`
+
+ // included is a slice of namespaced-scoped resource type names to include in
+ // the kubernetes resources.
+ // The default value is "*", which means all resource types will be included.
+ // +optional
+ // +kubebuilder:default={"*"}
+ Included []string `json:"included,omitempty"`
- // secret is used to connect to the target database cluster.
- // If not set, secret will be inherited from backup policy template.
- // if still not set, the controller will check if any system account for dataprotection has been created.
+ // excluded is a slice of namespaced-scoped resource type names to exclude in
+ // the kubernetes resources.
+ // The default value is empty.
// +optional
- Secret *BackupPolicySecret `json:"secret,omitempty"`
+ Excluded []string `json:"excluded,omitempty"`
}
-// BackupPolicySecret defines for the target database secret that backup tool can connect.
-type BackupPolicySecret struct {
- // the secret name
+// BackupMethod defines the backup method.
+type BackupMethod struct {
+ // the name of backup method.
// +kubebuilder:validation:Required
// +kubebuilder:validation:Pattern:=`^[a-z0-9]([a-z0-9\.\-]*[a-z0-9])?$`
Name string `json:"name"`
- // usernameKey the map key of the user in the connection credential secret
- // +kubebuilder:validation:Required
- // +kubebuilder:default=username
- UsernameKey string `json:"usernameKey,omitempty"`
-
- // passwordKey the map key of the password in the connection credential secret
- // +kubebuilder:validation:Required
- // +kubebuilder:default=password
- PasswordKey string `json:"passwordKey,omitempty"`
-}
-
-// BackupPolicyHook defines for the database execute commands before and after backup.
-type BackupPolicyHook struct {
- // pre backup to perform commands
+ // snapshotVolumes specifies whether to take snapshots of persistent volumes.
+ // if true, the BackupScript is not required, the controller will use the CSI
+ // volume snapshotter to create the snapshot.
// +optional
- PreCommands []string `json:"preCommands,omitempty"`
+ // +kubebuilder:default=false
+ SnapshotVolumes *bool `json:"snapshotVolumes,omitempty"`
- // post backup to perform commands
+ // actionSetName refers to the ActionSet object that defines the backup actions.
+ // For volume snapshot backup, the actionSet is not required, the controller
+ // will use the CSI volume snapshotter to create the snapshot.
// +optional
- PostCommands []string `json:"postCommands,omitempty"`
+ ActionSetName string `json:"actionSetName,omitempty"`
- // exec command with image
+ // targetVolumes specifies which volumes from the target should be mounted in
+ // the backup workload.
// +optional
- Image string `json:"image,omitempty"`
+ TargetVolumes *TargetVolumeInfo `json:"targetVolumes,omitempty"`
- // which container can exec command
+ // env specifies the environment variables for the backup workload.
// +optional
- ContainerName string `json:"containerName,omitempty"`
-}
-
-// BackupStatusUpdateStage defines the stage of backup status update.
-// +enum
-// +kubebuilder:validation:Enum={pre,post}
-type BackupStatusUpdateStage string
+ Env []corev1.EnvVar `json:"env,omitempty"`
-const (
- PRE BackupStatusUpdateStage = "pre"
- POST BackupStatusUpdateStage = "post"
-)
-
-type BackupStatusUpdate struct {
- // specify the json path of backup object for patch.
- // example: manifests.backupLog -- means patch the backup json path of status.manifests.backupLog.
+ // runtimeSettings specifies runtime settings for the backup workload container.
// +optional
- Path string `json:"path,omitempty"`
+ RuntimeSettings *RuntimeSettings `json:"runtimeSettings,omitempty"`
+}
- // which container name that kubectl can execute.
+// TargetVolumeInfo specifies the volumes and their mounts of the targeted application
+// that should be mounted in backup workload.
+type TargetVolumeInfo struct {
+ // Volumes indicates the list of volumes of targeted application that should
+ // be mounted on the backup job.
// +optional
- ContainerName string `json:"containerName,omitempty"`
+ Volumes []string `json:"volumes,omitempty"`
- // the shell Script commands to collect backup status metadata.
- // The script must exist in the container of ContainerName and the output format must be set to JSON.
- // Note that outputting to stderr may cause the result format to not be in JSON.
+ // volumeMounts specifies the mount for the volumes specified in `Volumes` section.
// +optional
- Script string `json:"script,omitempty"`
+ VolumeMounts []corev1.VolumeMount `json:"volumeMounts,omitempty"`
+}
- // useTargetPodServiceAccount defines whether this job requires the service account of the backup target pod.
- // if true, will use the service account of the backup target pod. otherwise, will use the system service account.
+type RuntimeSettings struct {
+ // resources specifies the resource required by container.
+ // More info: https://kubernetes.io/docs/concepts/configuration/manage-compute-resources-container/
// +optional
- UseTargetPodServiceAccount bool `json:"useTargetPodServiceAccount,omitempty"`
-
- // when to update the backup status, pre: before backup, post: after backup
- // +kubebuilder:validation:Required
- UpdateStage BackupStatusUpdateStage `json:"updateStage"`
+ Resources corev1.ResourceRequirements `json:"resources,omitempty"`
}
// BackupPolicyStatus defines the observed state of BackupPolicy
type BackupPolicyStatus struct {
-
- // observedGeneration is the most recent generation observed for this
- // BackupPolicy. It corresponds to the Cluster's generation, which is
- // updated on mutation by the API Server.
+ // phase - in list of [Available,Unavailable]
// +optional
- ObservedGeneration int64 `json:"observedGeneration,omitempty"`
+ Phase Phase `json:"phase,omitempty"`
- // backup policy phase valid value: Available, Failed.
+ // A human-readable message indicating details about why the BackupPolicy is
+ // in this phase.
// +optional
- Phase BackupPolicyPhase `json:"phase,omitempty"`
+ Message string `json:"message,omitempty"`
- // the reason if backup policy check failed.
+ // observedGeneration is the most recent generation observed for this
+ // BackupPolicy. It refers to the BackupPolicy's generation, which is
+ // updated on mutation by the API Server.
// +optional
- FailureReason string `json:"failureReason,omitempty"`
+ ObservedGeneration int64 `json:"observedGeneration,omitempty"`
+}
- // information when was the last time the job was successfully scheduled.
- // +optional
- LastScheduleTime *metav1.Time `json:"lastScheduleTime,omitempty"`
+// BackupPolicyPhase defines phases for BackupPolicy.
+// +enum
+// +kubebuilder:validation:Enum={Available,Failed}
+type BackupPolicyPhase string
- // information when was the last time the job successfully completed.
- // +optional
- LastSuccessfulTime *metav1.Time `json:"lastSuccessfulTime,omitempty"`
-}
+const (
+ BackupPolicyAvailable BackupPolicyPhase = "Available"
+ BackupPolicyFailed BackupPolicyPhase = "Failed"
+)
// +genclient
// +k8s:openapi-gen=true
// +kubebuilder:object:root=true
// +kubebuilder:subresource:status
// +kubebuilder:resource:categories={kubeblocks},scope=Namespaced,shortName=bp
+// +kubebuilder:printcolumn:name="BACKUP-REPO", type=string, JSONPath=`.spec.backupRepoName`
// +kubebuilder:printcolumn:name="STATUS",type=string,JSONPath=`.status.phase`
-// +kubebuilder:printcolumn:name="LAST SCHEDULE",type=string,JSONPath=`.status.lastScheduleTime`
// +kubebuilder:printcolumn:name="AGE",type=date,JSONPath=`.metadata.creationTimestamp`
-// BackupPolicy is the Schema for the backuppolicies API (defined by User)
+// BackupPolicy is the Schema for the backuppolicies API.
type BackupPolicy struct {
metav1.TypeMeta `json:",inline"`
metav1.ObjectMeta `json:"metadata,omitempty"`
@@ -323,53 +253,3 @@ type BackupPolicyList struct {
func init() {
SchemeBuilder.Register(&BackupPolicy{}, &BackupPolicyList{})
}
-
-func (r *BackupPolicySpec) GetCommonPolicy(backupType BackupType) *CommonBackupPolicy {
- switch backupType {
- case BackupTypeDataFile:
- return r.Datafile
- case BackupTypeLogFile:
- return r.Logfile
- }
- return nil
-}
-
-func (r *BackupPolicySpec) GetCommonSchedulePolicy(backupType BackupType) *SchedulePolicy {
- switch backupType {
- case BackupTypeSnapshot:
- return r.Schedule.Snapshot
- case BackupTypeDataFile:
- return r.Schedule.Datafile
- case BackupTypeLogFile:
- return r.Schedule.Logfile
- }
- return nil
-}
-
-// ToDuration converts the ttl string to time.Duration.
-func ToDuration(ttl *string) time.Duration {
- if ttl == nil {
- return time.Duration(0)
- }
- ttlLower := strings.ToLower(*ttl)
- if strings.HasSuffix(ttlLower, "d") {
- days, _ := strconv.Atoi(strings.ReplaceAll(ttlLower, "d", ""))
- return time.Hour * 24 * time.Duration(days)
- }
- hours, _ := strconv.Atoi(strings.ReplaceAll(ttlLower, "h", ""))
- return time.Hour * time.Duration(hours)
-}
-
-// AddTTL adds tll with hours
-func AddTTL(ttl *string, hours int) string {
- if ttl == nil {
- return ""
- }
- ttlLower := strings.ToLower(*ttl)
- if strings.HasSuffix(ttlLower, "d") {
- days, _ := strconv.Atoi(strings.ReplaceAll(ttlLower, "d", ""))
- return fmt.Sprintf("%dh", days*24+hours)
- }
- ttlHours, _ := strconv.Atoi(strings.ReplaceAll(ttlLower, "h", ""))
- return fmt.Sprintf("%dh", ttlHours+hours)
-}
diff --git a/apis/dataprotection/v1alpha1/backuppolicy_types_test.go b/apis/dataprotection/v1alpha1/backuppolicy_types_test.go
deleted file mode 100644
index b63d26e7b49..00000000000
--- a/apis/dataprotection/v1alpha1/backuppolicy_types_test.go
+++ /dev/null
@@ -1,83 +0,0 @@
-/*
-Copyright (C) 2022-2023 ApeCloud Co., Ltd
-
-Licensed under the Apache License, Version 2.0 (the "License");
-you may not use this file except in compliance with the License.
-You may obtain a copy of the License at
-
- http://www.apache.org/licenses/LICENSE-2.0
-
-Unless required by applicable law or agreed to in writing, software
-distributed under the License is distributed on an "AS IS" BASIS,
-WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
-See the License for the specific language governing permissions and
-limitations under the License.
-*/
-
-package v1alpha1
-
-import (
- "testing"
- "time"
-
- . "github.com/onsi/gomega"
-)
-
-func expectToDuration(t *testing.T, ttl string, baseNum, targetNum int) {
- d := ToDuration(&ttl)
- if d != time.Hour*time.Duration(baseNum)*time.Duration(targetNum) {
- t.Errorf(`Expected duration is "%d*%d*time.Hour"", got %v`, targetNum, baseNum, d)
- }
-}
-
-func TestToDuration(t *testing.T) {
- d := ToDuration(nil)
- if d != time.Duration(0) {
- t.Errorf("Expected duration is 0, got %v", d)
- }
- expectToDuration(t, "7d", 24, 7)
- expectToDuration(t, "7D", 24, 7)
- expectToDuration(t, "12h", 1, 12)
- expectToDuration(t, "12H", 1, 12)
-}
-
-func TestAddTTL(t *testing.T) {
- ttl := "7d"
- newTTL := AddTTL(&ttl, 12)
- if newTTL != "180h" {
- t.Errorf("expected new ttl is 180h, bur got %s", newTTL)
- }
- ttl = "7h"
- newTTL = AddTTL(&ttl, 12)
- if newTTL != "19h" {
- t.Errorf("expected new ttl is 19h, bur got %s", newTTL)
- }
-}
-
-func TestGetCommonPolicy(t *testing.T) {
- g := NewGomegaWithT(t)
-
- policySpec := &BackupPolicySpec{}
- g.Expect(policySpec.GetCommonPolicy(BackupTypeSnapshot)).Should(BeNil())
-
- policySpec = &BackupPolicySpec{Datafile: &CommonBackupPolicy{}, Logfile: &CommonBackupPolicy{}}
- g.Expect(policySpec.GetCommonPolicy(BackupTypeSnapshot)).Should(BeNil())
- g.Expect(policySpec.GetCommonPolicy(BackupTypeDataFile)).ShouldNot(BeNil())
- g.Expect(policySpec.GetCommonPolicy(BackupTypeLogFile)).ShouldNot(BeNil())
-}
-
-func TestGetCommonSchedulePolicy(t *testing.T) {
- g := NewGomegaWithT(t)
-
- policySpec := &BackupPolicySpec{}
- g.Expect(policySpec.GetCommonSchedulePolicy(BackupTypeSnapshot)).Should(BeNil())
-
- policySpec = &BackupPolicySpec{Schedule: Schedule{
- Snapshot: &SchedulePolicy{},
- Datafile: &SchedulePolicy{},
- Logfile: &SchedulePolicy{},
- }}
- g.Expect(policySpec.GetCommonSchedulePolicy(BackupTypeSnapshot)).ShouldNot(BeNil())
- g.Expect(policySpec.GetCommonSchedulePolicy(BackupTypeDataFile)).ShouldNot(BeNil())
- g.Expect(policySpec.GetCommonSchedulePolicy(BackupTypeLogFile)).ShouldNot(BeNil())
-}
diff --git a/apis/dataprotection/v1alpha1/backuprepo_types.go b/apis/dataprotection/v1alpha1/backuprepo_types.go
index 7266218edd3..f89c6156e35 100644
--- a/apis/dataprotection/v1alpha1/backuprepo_types.go
+++ b/apis/dataprotection/v1alpha1/backuprepo_types.go
@@ -22,6 +22,18 @@ import (
metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
)
+// AccessMethod is an enum type that defines the access method of the backup repo.
+type AccessMethod string
+
+const (
+ // AccessMethodMount means that the storage is mounted locally,
+ // so that remote files can be accessed just like a local file.
+ AccessMethodMount AccessMethod = "Mount"
+ // AccessMethodTool means to access the storage with a command-line tool,
+ // which helps to transfer files between the storage and local.
+ AccessMethodTool AccessMethod = "Tool"
+)
+
// BackupRepoSpec defines the desired state of BackupRepo
type BackupRepoSpec struct {
// The storage provider used by this backup repo.
@@ -29,6 +41,12 @@ type BackupRepoSpec struct {
// +kubebuilder:validation:Required
StorageProviderRef string `json:"storageProviderRef"`
+ // Specifies the access method of the backup repo.
+ // +kubebuilder:validation:Enum={Mount,Tool}
+ // +kubebuilder:default=Mount
+ // +optional
+ AccessMethod AccessMethod `json:"accessMethod,omitempty"`
+
// The requested capacity for the PVC created by this backup repo.
// +optional
VolumeCapacity resource.Quantity `json:"volumeCapacity,omitempty"`
@@ -74,6 +92,10 @@ type BackupRepoStatus struct {
// +optional
BackupPVCName string `json:"backupPVCName,omitempty"`
+ // toolConfigSecretName is the name of the secret containing the configuration for the access tool.
+ // +optional
+ ToolConfigSecretName string `json:"toolConfigSecretName,omitempty"`
+
// isDefault indicates whether this backup repo is the default one.
// +optional
IsDefault bool `json:"isDefault,omitempty"`
@@ -111,3 +133,11 @@ type BackupRepoList struct {
func init() {
SchemeBuilder.Register(&BackupRepo{}, &BackupRepoList{})
}
+
+func (repo *BackupRepo) AccessByMount() bool {
+ return repo.Spec.AccessMethod == "" || repo.Spec.AccessMethod == AccessMethodMount
+}
+
+func (repo *BackupRepo) AccessByTool() bool {
+ return repo.Spec.AccessMethod == AccessMethodTool
+}
diff --git a/apis/dataprotection/v1alpha1/backupschedule_types.go b/apis/dataprotection/v1alpha1/backupschedule_types.go
new file mode 100644
index 00000000000..a707a2fb167
--- /dev/null
+++ b/apis/dataprotection/v1alpha1/backupschedule_types.go
@@ -0,0 +1,159 @@
+/*
+Copyright (C) 2022-2023 ApeCloud Co., Ltd
+
+Licensed under the Apache License, Version 2.0 (the "License");
+you may not use this file except in compliance with the License.
+You may obtain a copy of the License at
+
+ http://www.apache.org/licenses/LICENSE-2.0
+
+Unless required by applicable law or agreed to in writing, software
+distributed under the License is distributed on an "AS IS" BASIS,
+WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+See the License for the specific language governing permissions and
+limitations under the License.
+*/
+
+package v1alpha1
+
+import (
+ metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
+)
+
+// BackupScheduleSpec defines the desired state of BackupSchedule.
+type BackupScheduleSpec struct {
+ // Which backupPolicy is applied to perform this backup.
+ // +kubebuilder:validation:Required
+ // +kubebuilder:validation:Pattern:=`^[a-z0-9]([a-z0-9\.\-]*[a-z0-9])?$`
+ BackupPolicyName string `json:"backupPolicyName"`
+
+ // startingDeadlineMinutes defines the deadline in minutes for starting the
+ // backup workload if it misses scheduled time for any reason.
+ // +optional
+ // +kubebuilder:validation:Minimum=0
+ // +kubebuilder:validation:Maximum=1440
+ StartingDeadlineMinutes *int64 `json:"startingDeadlineMinutes,omitempty"`
+
+ // schedules defines the list of backup schedules.
+ // +kubebuilder:validation:Required
+ // +kubebuilder:validation:MinItems=1
+ Schedules []SchedulePolicy `json:"schedules"`
+}
+
+type SchedulePolicy struct {
+ // enabled specifies whether the backup schedule is enabled or not.
+ // +optional
+ Enabled *bool `json:"enabled,omitempty"`
+
+ // backupMethod specifies the backup method name that is defined in backupPolicy.
+ // +kubebuilder:validation:Required
+ BackupMethod string `json:"backupMethod"`
+
+ // the cron expression for schedule, the timezone is in UTC.
+ // see https://en.wikipedia.org/wiki/Cron.
+ // +kubebuilder:validation:Required
+ CronExpression string `json:"cronExpression"`
+
+ // retentionPeriod determines a duration up to which the backup should be kept.
+ // controller will remove all backups that are older than the RetentionPeriod.
+ // For example, RetentionPeriod of `30d` will keep only the backups of last 30 days.
+ // Sample duration format:
+ // - years: 2y
+ // - months: 6mo
+ // - days: 30d
+ // - hours: 12h
+ // - minutes: 30m
+ // You can also combine the above durations. For example: 30d12h30m
+ // +optional
+ // +kubebuilder:default="7d"
+ RetentionPeriod RetentionPeriod `json:"retentionPeriod,omitempty"`
+}
+
+// BackupScheduleStatus defines the observed state of BackupSchedule.
+type BackupScheduleStatus struct {
+ // phase describes the phase of the BackupSchedule.
+ // +optional
+ Phase BackupSchedulePhase `json:"phase,omitempty"`
+
+ // observedGeneration is the most recent generation observed for this
+ // BackupSchedule. It refers to the BackupSchedule's generation, which is
+ // updated on mutation by the API Server.
+ // +optional
+ ObservedGeneration int64 `json:"observedGeneration,omitempty"`
+
+ // failureReason is an error that caused the backup to fail.
+ // +optional
+ FailureReason string `json:"failureReason,omitempty"`
+
+ // schedules describes the status of each schedule.
+ // +optional
+ Schedules map[string]ScheduleStatus `json:"schedules,omitempty"`
+}
+
+// BackupSchedulePhase defines the phase of BackupSchedule
+type BackupSchedulePhase string
+
+const (
+ // BackupSchedulePhaseAvailable means the backup schedule is available.
+ BackupSchedulePhaseAvailable BackupSchedulePhase = "Available"
+
+ // BackupSchedulePhaseFailed means the backup schedule is failed.
+ BackupSchedulePhaseFailed BackupSchedulePhase = "Failed"
+)
+
+// ScheduleStatus defines the status of each schedule.
+type ScheduleStatus struct {
+ // phase describes the phase of the schedule.
+ // +optional
+ Phase SchedulePhase `json:"phase,omitempty"`
+
+ // failureReason is an error that caused the backup to fail.
+ // +optional
+ FailureReason string `json:"failureReason,omitempty"`
+
+ // lastScheduleTime records the last time the backup was scheduled.
+ // +optional
+ LastScheduleTime *metav1.Time `json:"lastScheduleTime,omitempty"`
+
+ // lastSuccessfulTime records the last time the backup was successfully completed.
+ // +optional
+ LastSuccessfulTime *metav1.Time `json:"lastSuccessfulTime,omitempty"`
+}
+
+// SchedulePhase defines the phase of schedule
+type SchedulePhase string
+
+const (
+ ScheduleRunning SchedulePhase = "Running"
+ ScheduleFailed SchedulePhase = "Failed"
+)
+
+// +genclient
+// +k8s:openapi-gen=true
+// +kubebuilder:object:root=true
+// +kubebuilder:subresource:status
+// +kubebuilder:resource:categories={kubeblocks},scope=Namespaced,shortName=bs
+// +kubebuilder:printcolumn:name="STATUS",type=string,JSONPath=`.status.phase`
+// +kubebuilder:printcolumn:name="AGE",type=date,JSONPath=`.metadata.creationTimestamp`
+
+// BackupSchedule is the Schema for the backupschedules API.
+type BackupSchedule struct {
+ metav1.TypeMeta `json:",inline"`
+ metav1.ObjectMeta `json:"metadata,omitempty"`
+
+ Spec BackupScheduleSpec `json:"spec,omitempty"`
+ Status BackupScheduleStatus `json:"status,omitempty"`
+}
+
+// +kubebuilder:object:root=true
+
+// BackupScheduleList contains a list of BackupSchedule.
+type BackupScheduleList struct {
+ metav1.TypeMeta `json:",inline"`
+ metav1.ListMeta `json:"metadata,omitempty"`
+ Items []BackupSchedule `json:"items"`
+}
+
+func init() {
+ SchemeBuilder.Register(&BackupSchedule{}, &BackupScheduleList{})
+}
diff --git a/apis/dataprotection/v1alpha1/backuptool_types.go b/apis/dataprotection/v1alpha1/backuptool_types.go
deleted file mode 100644
index c4e82a2797b..00000000000
--- a/apis/dataprotection/v1alpha1/backuptool_types.go
+++ /dev/null
@@ -1,161 +0,0 @@
-/*
-Copyright (C) 2022-2023 ApeCloud Co., Ltd
-
-Licensed under the Apache License, Version 2.0 (the "License");
-you may not use this file except in compliance with the License.
-You may obtain a copy of the License at
-
- http://www.apache.org/licenses/LICENSE-2.0
-
-Unless required by applicable law or agreed to in writing, software
-distributed under the License is distributed on an "AS IS" BASIS,
-WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
-See the License for the specific language governing permissions and
-limitations under the License.
-*/
-
-package v1alpha1
-
-import (
- corev1 "k8s.io/api/core/v1"
- metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
-)
-
-// BackupToolSpec defines the desired state of BackupTool
-type BackupToolSpec struct {
- // Backup tool Container image name.
- // +kubebuilder:validation:Required
- Image string `json:"image"`
-
- // which kind for run a backup tool, supported values: job, statefulSet.
- // +kubebuilder:default=job
- DeployKind DeployKind `json:"deployKind,omitempty"`
-
- // the type of backup tool, file or pitr
- // +kubebuilder:validation:Enum={file,pitr}
- // +kubebuilder:default=file
- Type string `json:"type,omitempty"`
-
- // Compute Resources required by this container.
- // Cannot be updated.
- // +kubebuilder:pruning:PreserveUnknownFields
- // +optional
- Resources *corev1.ResourceRequirements `json:"resources,omitempty"`
-
- // List of environment variables to set in the container.
- // +kubebuilder:pruning:PreserveUnknownFields
- // +optional
- Env []corev1.EnvVar `json:"env,omitempty" patchStrategy:"merge" patchMergeKey:"name"`
-
- // List of sources to populate environment variables in the container.
- // The keys defined within a source must be a C_IDENTIFIER. All invalid keys
- // will be reported as an event when the container is starting. When a key exists in multiple
- // sources, the value associated with the last source will take precedence.
- // Values defined by an Env with a duplicate key will take precedence.
- // Cannot be updated.
- // +kubebuilder:pruning:PreserveUnknownFields
- // +optional
- EnvFrom []corev1.EnvFromSource `json:"envFrom,omitempty"`
-
- // Array of command that apps can do database backup.
- // from invoke args
- // the order of commands follows the order of array.
- // +kubebuilder:validation:Required
- BackupCommands []string `json:"backupCommands"`
-
- // Array of command that apps can do database incremental backup.
- // like xtrabackup, that can performs an incremental backup file.
- // +optional
- IncrementalBackupCommands []string `json:"incrementalBackupCommands,omitempty"`
-
- // backup tool can support physical restore, in this case, restore must be RESTART database.
- // +optional
- Physical *PhysicalConfig `json:"physical,omitempty"`
-
- // backup tool can support logical restore, in this case, restore NOT RESTART database.
- // +optional
- Logical *LogicalConfig `json:"logical,omitempty"`
-}
-
-type LogicalConfig struct {
- BackupToolRestoreCommand `json:",inline"`
-
- // podScope defines the pod scope for restore from backup, supported values:
- // - 'All' will exec the restore command on all pods.
- // - 'ReadWrite' will pick a ReadWrite pod to exec the restore command.
- // +optional
- // +kubebuilder:default=All
- PodScope PodRestoreScope `json:"podScope,omitempty"`
-}
-
-type PhysicalConfig struct {
- BackupToolRestoreCommand `json:",inline"`
-
- // relyOnLogfile defines whether the current recovery relies on log files
- // +optional
- RelyOnLogfile bool `json:"relyOnLogfile,omitempty"`
-}
-
-// BackupToolRestoreCommand defines the restore commands of BackupTool
-type BackupToolRestoreCommand struct {
- // Array of command that apps can perform database restore.
- // like xtrabackup, that can performs restore mysql from files.
- // +optional
- RestoreCommands []string `json:"restoreCommands"`
-
- // Array of incremental restore commands.
- // +optional
- IncrementalRestoreCommands []string `json:"incrementalRestoreCommands,omitempty"`
-}
-
-// BackupToolStatus defines the observed state of BackupTool
-type BackupToolStatus struct {
- // TODO(dsj): define backup tool status.
-}
-
-// +genclient
-// +genclient:nonNamespaced
-// +k8s:openapi-gen=true
-// +kubebuilder:object:root=true
-// +kubebuilder:subresource:status
-// +kubebuilder:resource:categories={kubeblocks},scope=Cluster
-
-// BackupTool is the Schema for the backuptools API (defined by provider)
-type BackupTool struct {
- metav1.TypeMeta `json:",inline"`
- metav1.ObjectMeta `json:"metadata,omitempty"`
-
- Spec BackupToolSpec `json:"spec,omitempty"`
- Status BackupToolStatus `json:"status,omitempty"`
-}
-
-// +kubebuilder:object:root=true
-
-// BackupToolList contains a list of BackupTool
-type BackupToolList struct {
- metav1.TypeMeta `json:",inline"`
- metav1.ListMeta `json:"metadata,omitempty"`
- Items []BackupTool `json:"items"`
-}
-
-func init() {
- SchemeBuilder.Register(&BackupTool{}, &BackupToolList{})
-}
-
-func (physical *PhysicalConfig) GetPhysicalRestoreCommand() []string {
- if physical == nil || len(physical.RestoreCommands) == 0 {
- return nil
- }
- return physical.RestoreCommands
-}
-
-func (physical *PhysicalConfig) IsRelyOnLogfile() bool {
- return physical != nil && physical.RelyOnLogfile
-}
-
-func (logical *LogicalConfig) GetLogicalRestoreCommand() []string {
- if logical == nil || len(logical.RestoreCommands) == 0 {
- return nil
- }
- return logical.RestoreCommands
-}
diff --git a/apis/dataprotection/v1alpha1/restore_types.go b/apis/dataprotection/v1alpha1/restore_types.go
new file mode 100644
index 00000000000..fa9d110a1e5
--- /dev/null
+++ b/apis/dataprotection/v1alpha1/restore_types.go
@@ -0,0 +1,438 @@
+/*
+Copyright (C) 2022-2023 ApeCloud Co., Ltd
+
+This file is part of KubeBlocks project
+
+This program is free software: you can redistribute it and/or modify
+it under the terms of the GNU Affero General Public License as published by
+the Free Software Foundation, either version 3 of the License, or
+(at your option) any later version.
+
+This program is distributed in the hope that it will be useful
+but WITHOUT ANY WARRANTY; without even the implied warranty of
+MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+GNU Affero General Public License for more details.
+
+You should have received a copy of the GNU Affero General Public License
+along with this program. If not, see .
+*/
+
+package v1alpha1
+
+import (
+ corev1 "k8s.io/api/core/v1"
+ metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
+)
+
+// RestoreSpec defines the desired state of Restore
+type RestoreSpec struct {
+ // backup name, the following behavior based on the backup type:
+ // 1. Full: will be restored the full backup directly.
+ // 2. Incremental: will be restored sequentially from the most recent full backup of this incremental backup.
+ // 3. Differential: will be restored sequentially from the parent backup of the differential backup.
+ // 4. Continuous: will find the most recent full backup at this time point and the input continuous backup to restore.
+ // +kubebuilder:validation:Required
+ // +kubebuilder:validation:XValidation:rule="self == oldSelf",message="forbidden to update spec.backupName"
+ Backup BackupConfig `json:"backup"`
+
+ // restore according to a specified point in time.
+ // +kubebuilder:validation:XValidation:rule="self == oldSelf",message="forbidden to update spec.restoreTime"
+ // +optional
+ // +kubebuilder:validation:Pattern=`^\d{4}-\d{2}-\d{2}T\d{2}:\d{2}:\d{2}Z$`
+ RestoreTime string `json:"restoreTime,omitempty"`
+
+ // restore the specified resources of kubernetes.
+ // +kubebuilder:validation:XValidation:rule="self == oldSelf",message="forbidden to update spec.resources"
+ // +optional
+ Resources *RestoreKubeResources `json:"resources,omitempty"`
+
+ // configuration for the action of "prepareData" phase, including the persistent volume claims
+ // that need to be restored and scheduling strategy of temporary recovery pod.
+ // +optional
+ PrepareDataConfig *PrepareDataConfig `json:"prepareDataConfig,omitempty"`
+
+ // service account name which needs for recovery pod.
+ // +optional
+ ServiceAccountName string `json:"serviceAccountName,omitempty"`
+
+ // configuration for the action of "postReady" phase.
+ // +kubebuilder:validation:XValidation:rule="has(self.jobAction) || has(self.execAction)", message="at least one exists for jobAction and execAction."
+ // +optional
+ ReadyConfig *ReadyConfig `json:"readyConfig,omitempty"`
+
+ // list of environment variables to set in the container for restore and will be merged with the env of Backup and ActionSet.
+ // the priority of merging is as follows: Restore env > Backup env > ActionSet env.
+ // +kubebuilder:pruning:PreserveUnknownFields
+ // +optional
+ Env []corev1.EnvVar `json:"env,omitempty" patchStrategy:"merge" patchMergeKey:"name"`
+
+ // specified the required resources of restore job's container.
+ // +optional
+ ContainerResources corev1.ResourceRequirements `json:"containerResources,omitempty"`
+}
+
+type BackupConfig struct {
+ // backup name
+ // +kubebuilder:validation:Required
+ Name string `json:"name"`
+
+ // backup namespace
+ // +kubebuilder:validation:Required
+ Namespace string `json:"namespace"`
+}
+
+type RestoreKubeResources struct {
+ // will restore the specified resources
+ IncludeResources []IncludeResource `json:"included,omitempty"`
+
+ // TODO: supports exclude resources for recovery
+}
+
+type IncludeResource struct {
+ //
+ // +kubebuilder:validation:Required
+ GroupResource string `json:"groupResource"`
+
+ // select the specified resource for recovery by label.
+ // +optional
+ LabelSelector metav1.LabelSelector `json:"labelSelector,omitempty"`
+}
+
+type PrepareDataConfig struct {
+ // dataSourceRef describes the configuration when using `persistentVolumeClaim.spec.dataSourceRef` method for restoring.
+ // it describes the source volume of the backup targetVolumes and how to mount path in the restoring container.
+ // +kubebuilder:validation:XValidation:rule="self.volumeSource != '' || self.mountPath !=''",message="at least one exists for volumeSource and mountPath."
+ // +kubebuilder:validation:XValidation:rule="self == oldSelf",message="forbidden to update spec.prepareDataConfig.dataSourceRef"
+ // +optional
+ DataSourceRef *VolumeConfig `json:"dataSourceRef,omitempty"`
+
+ // volumeClaims defines the persistent Volume claims that need to be restored and mount them together into the restore job.
+ // these persistent Volume claims will be created if not exist.
+ // +kubebuilder:validation:XValidation:rule="self == oldSelf",message="forbidden to update spec.prepareDataConfig.volumeClaims"
+ // +patchMergeKey=name
+ // +patchStrategy=merge,retainKeys
+ // +optional
+ RestoreVolumeClaims []RestoreVolumeClaim `json:"volumeClaims,omitempty"`
+
+ // volumeClaimsTemplate defines a template to build persistent Volume claims that need to be restored.
+ // these claims will be created in an orderly manner based on the number of replicas or reused if already exist.
+ // +kubebuilder:validation:XValidation:rule="self == oldSelf",message="forbidden to update spec.prepareDataConfig.volumeClaimsTemplate"
+ // +patchMergeKey=name
+ // +patchStrategy=merge,retainKeys
+ // +optional
+ RestoreVolumeClaimsTemplate *RestoreVolumeClaimsTemplate `json:"volumeClaimsTemplate,omitempty"`
+
+ // VolumeClaimManagementPolicy defines recovery strategy for persistent volume claim. supported policies are as follows:
+ // 1. Parallel: parallel recovery of persistent volume claim.
+ // 2. Serial: restore the persistent volume claim in sequence, and wait until the previous persistent volume claim is restored before restoring a new one.
+ // +kubebuilder:default=Parallel
+ // +kubebuilder:validation:Required
+ VolumeClaimManagementPolicy VolumeClaimManagementPolicy `json:"volumeClaimManagementPolicy"`
+
+ // scheduling spec for restoring pod.
+ // +kubebuilder:validation:XValidation:rule="self == oldSelf",message="forbidden to update spec.prepareDataConfig.schedulingSpec"
+ // +optional
+ SchedulingSpec SchedulingSpec `json:"schedulingSpec,omitempty"`
+}
+
+type ReadyConfig struct {
+ // configuration for job action.
+ // +optional
+ JobAction *JobAction `json:"jobAction,omitempty"`
+
+ // configuration for exec action.
+ // +optional
+ ExecAction *ExecAction `json:"execAction,omitempty"`
+
+ // credential template used for creating a connection credential
+ // +optional
+ ConnectCredential *ConnectCredential `json:"connectCredential,omitempty"`
+
+ // periodic probe of the service readiness.
+ // controller will perform postReadyHooks of BackupScript.spec.restore after the service readiness when readinessProbe is configured.
+ // +optional
+ ReadinessProbe *ReadinessProbe `json:"readinessProbe,omitempty"`
+}
+
+type JobAction struct {
+ // jobActionTarget defines the pod that need to be executed for the job action.
+ // will select a pod that meets the conditions to execute.
+ // +kubebuilder:validation:Required
+ Target JobActionTarget `json:"target"`
+}
+
+type ExecAction struct {
+ // execActionTarget defines the pods that need to be executed for the exec action.
+ // will execute on all pods that meet the conditions.
+ // +optional
+ Target ExecActionTarget `json:"target"`
+}
+
+type ExecActionTarget struct {
+ // kubectl exec in all selected pods.
+ // +kubebuilder:validation:Required
+ PodSelector metav1.LabelSelector `json:"podSelector"`
+}
+
+type JobActionTarget struct {
+ // select one of the pods which selected by labels to build the job spec, such as mount required volumes and inject built-in env of the selected pod.
+ // +kubebuilder:validation:Required
+ PodSelector metav1.LabelSelector `json:"podSelector"`
+
+ // volumeMounts defines which volumes of the selected pod need to be mounted on the restoring pod.
+ // +patchMergeKey=name
+ // +patchStrategy=merge,retainKeys
+ // +optional
+ VolumeMounts []corev1.VolumeMount `json:"volumeMounts,omitempty"`
+}
+
+type VolumeConfig struct {
+ // volumeSource describes the volume will be restored from the specified volume of the backup targetVolumes.
+ // required if the backup uses volume snapshot.
+ // +optional
+ VolumeSource string `json:"volumeSource,omitempty"`
+
+ // mountPath path within the restoring container at which the volume should be mounted.
+ // +optional
+ MountPath string `json:"mountPath,omitempty"`
+}
+
+type RestoreVolumeClaim struct {
+ // Standard object's metadata.
+ // More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#metadata
+ // +kubebuilder:validation:Required
+ metav1.ObjectMeta `json:"metadata"`
+
+ // volumeClaimSpec defines the desired characteristics of a persistent volume claim.
+ // +kubebuilder:validation:Required
+ VolumeClaimSpec corev1.PersistentVolumeClaimSpec `json:"volumeClaimSpec"`
+
+ // describing the source volume of the backup targetVolumes and how to mount path in the restoring container.
+ // +kubebuilder:validation:XValidation:rule="self.volumeSource != '' || self.mountPath !=''",message="at least one exists for volumeSource and mountPath."
+ VolumeConfig `json:",inline"`
+}
+
+type RestoreVolumeClaimsTemplate struct {
+ // templates is a list of volume claims.
+ // +kubebuilder:validation:Required
+ Templates []RestoreVolumeClaim `json:"templates"`
+
+ // the replicas of persistent volume claim which need to be created and restored.
+ // the format of created claim name is "-".
+ // +kubebuilder:validation:Minimum=1
+ // +kubebuilder:validation:Required
+ Replicas int32 `json:"replicas"`
+
+ // the starting index for the created persistent volume claim by according to template.
+ // minimum is 0.
+ // +kubebuilder:validation:Minimum=0
+ StartingIndex int32 `json:"startingIndex,omitempty"`
+}
+
+type SchedulingSpec struct {
+ // the restoring pod's tolerations.
+ // +optional
+ Tolerations []corev1.Toleration `json:"tolerations,omitempty"`
+
+ // nodeSelector is a selector which must be true for the pod to fit on a node.
+ // Selector which must match a node's labels for the pod to be scheduled on that node.
+ // More info: https://kubernetes.io/docs/concepts/configuration/assign-pod-node/
+ // +optional
+ // +mapType=atomic
+ NodeSelector map[string]string `json:"nodeSelector,omitempty"`
+
+ // nodeName is a request to schedule this pod onto a specific node. If it is non-empty,
+ // the scheduler simply schedules this pod onto that node, assuming that it fits resource
+ // requirements.
+ // +optional
+ NodeName string `json:"nodeName,omitempty"`
+
+ // affinity is a group of affinity scheduling rules.
+ // refer to https://kubernetes.io/docs/concepts/configuration/assign-pod-node/
+ // +optional
+ Affinity *corev1.Affinity `json:"affinity,omitempty"`
+
+ // topologySpreadConstraints describes how a group of pods ought to spread across topology
+ // domains. Scheduler will schedule pods in a way which abides by the constraints.
+ // refer to https://kubernetes.io/docs/concepts/scheduling-eviction/topology-spread-constraints/
+ // +optional
+ TopologySpreadConstraints []corev1.TopologySpreadConstraint `json:"topologySpreadConstraints,omitempty"`
+
+ // If specified, the pod will be dispatched by specified scheduler.
+ // If not specified, the pod will be dispatched by default scheduler.
+ // +optional
+ SchedulerName string `json:"schedulerName,omitempty"`
+}
+
+type ConnectCredential struct {
+ // the secret name
+ // +kubebuilder:validation:Required
+ // +kubebuilder:validation:Pattern:=`^[a-z0-9]([a-z0-9\.\-]*[a-z0-9])?$`
+ SecretName string `json:"secretName"`
+
+ // usernameKey the map key of the user in the connection credential secret
+ // +kubebuilder:validation:Required
+ // +kubebuilder:default=username
+ UsernameKey string `json:"usernameKey"`
+
+ // passwordKey the map key of the password in the connection credential secret
+ // +kubebuilder:validation:Required
+ // +kubebuilder:default=password
+ PasswordKey string `json:"passwordKey"`
+
+ // hostKey the map key of the host in the connection credential secret
+ // +kubebuilder:default=host
+ HostKey string `json:"hostKey,omitempty"`
+
+ // portKey the map key of the port in the connection credential secret
+ // +kubebuilder:default=port
+ PortKey string `json:"portKey,omitempty"`
+}
+
+type ReadinessProbe struct {
+ // number of seconds after the container has started before probe is initiated.
+ // +optional
+ // +kubebuilder:validation:Minimum=0
+ InitialDelaySeconds int `json:"initialDelaySeconds,omitempty"`
+
+ // number of seconds after which the probe times out.
+ // defaults to 30 second, minimum value is 1.
+ // +optional
+ // +kubebuilder:default=30
+ // +kubebuilder:validation:Minimum=1
+ TimeoutSeconds int `json:"timeoutSeconds"`
+
+ // how often (in seconds) to perform the probe.
+ // defaults to 5 second, minimum value is 1.
+ // +optional
+ // +kubebuilder:default=5
+ // +kubebuilder:validation:Minimum=1
+ PeriodSeconds int `json:"periodSeconds"`
+
+ // exec specifies the action to take.
+ // +kubebuilder:validation:Required
+ Exec ReadinessProbeExecAction `json:"exec"`
+
+ // TODO: support readiness probe by checking k8s resource
+}
+
+type ReadinessProbeExecAction struct {
+ // refer to container image.
+ // +kubebuilder:validation:Required
+ Image string `json:"image"`
+
+ // refer to container command.
+ // +kubebuilder:validation:Required
+ Command []string `json:"command"`
+}
+
+type RestoreStatusActions struct {
+ // record the actions for prepareData phase.
+ // +patchMergeKey=jobName
+ // +patchStrategy=merge,retainKeys
+ // +optional
+ PrepareData []RestoreStatusAction `json:"prepareData,omitempty"`
+
+ // record the actions for postReady phase.
+ // +patchMergeKey=jobName
+ // +patchStrategy=merge,retainKeys
+ // +optional
+ PostReady []RestoreStatusAction `json:"postReady,omitempty"`
+}
+
+type RestoreStatusAction struct {
+ // name describes the name of the recovery action based on the current backup.
+ // +kubebuilder:validation:Required
+ Name string `json:"name"`
+
+ // which backup's restore action belongs to.
+ // +kubebuilder:validation:Required
+ BackupName string `json:"backupName"`
+
+ // the execution object of the restore action.
+ // +kubebuilder:validation:Required
+ ObjectKey string `json:"objectKey"`
+
+ // message is a human readable message indicating details about the object condition.
+ // +optional
+ Message string `json:"message,omitempty"`
+
+ // the status of this action.
+ // +kubebuilder:validation:Required
+ Status RestoreActionStatus `json:"status,omitempty"`
+
+ // startTime is the start time for the restore job.
+ // +optional
+ StartTime metav1.Time `json:"startTime,omitempty"`
+
+ // endTime is the completion time for the restore job.
+ // +optional
+ EndTime metav1.Time `json:"endTime,omitempty"`
+}
+
+// RestoreStatus defines the observed state of Restore
+type RestoreStatus struct {
+ // +optional
+ Phase RestorePhase `json:"phase,omitempty"`
+
+ // Date/time when the restore started being processed.
+ // +optional
+ StartTimestamp *metav1.Time `json:"startTimestamp,omitempty"`
+
+ // Date/time when the restore finished being processed.
+ // +optional
+ CompletionTimestamp *metav1.Time `json:"completionTimestamp,omitempty"`
+
+ // The duration time of restore execution.
+ // When converted to a string, the form is "1h2m0.5s".
+ // +optional
+ Duration *metav1.Duration `json:"duration,omitempty"`
+
+ // recorded all restore actions performed.
+ // +optional
+ Actions RestoreStatusActions `json:"actions,omitempty"`
+
+ // describe current state of restore API Resource, like warning.
+ // +optional
+ Conditions []metav1.Condition `json:"conditions,omitempty"`
+}
+
+// +genclient
+// +k8s:openapi-gen=true
+// +kubebuilder:object:root=true
+// +kubebuilder:subresource:status
+// +kubebuilder:resource:categories={kubeblocks,all}
+// +kubebuilder:printcolumn:name="BACKUP",type="string",JSONPath=".spec.backup.name"
+// +kubebuilder:printcolumn:name="RESTORE-TIME",type="string",JSONPath=".spec.restoreTime",description="Point in time for restoring"
+// +kubebuilder:printcolumn:name="STATUS",type="string",JSONPath=".status.phase",description="Restore Status."
+// +kubebuilder:printcolumn:name="DURATION",type=string,JSONPath=".status.duration"
+// +kubebuilder:printcolumn:name="CREATE-TIME",type=string,JSONPath=".metadata.creationTimestamp"
+// +kubebuilder:printcolumn:name="COMPLETION-TIME",type=string,JSONPath=".status.completionTimestamp"
+
+// Restore is the Schema for the restores API
+type Restore struct {
+ metav1.TypeMeta `json:",inline"`
+ metav1.ObjectMeta `json:"metadata,omitempty"`
+
+ Spec RestoreSpec `json:"spec,omitempty"`
+ Status RestoreStatus `json:"status,omitempty"`
+}
+
+//+kubebuilder:object:root=true
+
+// RestoreList contains a list of Restore
+type RestoreList struct {
+ metav1.TypeMeta `json:",inline"`
+ metav1.ListMeta `json:"metadata,omitempty"`
+ Items []Restore `json:"items"`
+}
+
+func init() {
+ SchemeBuilder.Register(&Restore{}, &RestoreList{})
+}
+func (p *PrepareDataConfig) IsSerialPolicy() bool {
+ if p == nil {
+ return false
+ }
+ return p.VolumeClaimManagementPolicy == SerialManagementPolicy
+}
diff --git a/apis/dataprotection/v1alpha1/restorejob_types.go b/apis/dataprotection/v1alpha1/restorejob_types.go
deleted file mode 100644
index a48c3f8afba..00000000000
--- a/apis/dataprotection/v1alpha1/restorejob_types.go
+++ /dev/null
@@ -1,102 +0,0 @@
-/*
-Copyright (C) 2022-2023 ApeCloud Co., Ltd
-
-Licensed under the Apache License, Version 2.0 (the "License");
-you may not use this file except in compliance with the License.
-You may obtain a copy of the License at
-
- http://www.apache.org/licenses/LICENSE-2.0
-
-Unless required by applicable law or agreed to in writing, software
-distributed under the License is distributed on an "AS IS" BASIS,
-WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
-See the License for the specific language governing permissions and
-limitations under the License.
-*/
-
-package v1alpha1
-
-import (
- corev1 "k8s.io/api/core/v1"
- metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
-)
-
-// RestoreJobSpec defines the desired state of RestoreJob
-type RestoreJobSpec struct {
- // Specified one backupJob to restore.
- // +kubebuilder:validation:Required
- BackupJobName string `json:"backupJobName"`
-
- // the target database workload to restore
- // +kubebuilder:validation:Required
- Target TargetCluster `json:"target"`
-
- // array of restore volumes .
- // +kubebuilder:validation:MinItems=1
- // +kubebuilder:pruning:PreserveUnknownFields
- TargetVolumes []corev1.Volume `json:"targetVolumes" patchStrategy:"merge,retainKeys" patchMergeKey:"name"`
-
- // array of restore volume mounts .
- // +kubebuilder:validation:MinItems=1
- // +kubebuilder:pruning:PreserveUnknownFields
- TargetVolumeMounts []corev1.VolumeMount `json:"targetVolumeMounts" patchStrategy:"merge" patchMergeKey:"mountPath"`
-
- // count of backup stop retries on fail.
- // +optional
- OnFailAttempted int32 `json:"onFailAttempted,omitempty"`
-}
-
-// RestoreJobStatus defines the observed state of RestoreJob
-type RestoreJobStatus struct {
-
- // +optional
- Phase RestoreJobPhase `json:"phase,omitempty"`
-
- // The date and time when the Backup is eligible for garbage collection.
- // 'null' means the Backup is NOT be cleaned except delete manual.
- // +optional
- Expiration *metav1.Time `json:"expiration,omitempty"`
-
- // Date/time when the backup started being processed.
- // +optional
- StartTimestamp *metav1.Time `json:"startTimestamp,omitempty"`
-
- // Date/time when the backup finished being processed.
- // +optional
- CompletionTimestamp *metav1.Time `json:"completionTimestamp,omitempty"`
-
- // Job failed reason.
- // +optional
- FailureReason string `json:"failureReason,omitempty"`
-}
-
-// +genclient
-// +k8s:openapi-gen=true
-// +kubebuilder:object:root=true
-// +kubebuilder:subresource:status
-// +kubebuilder:resource:categories={kubeblocks},scope=Namespaced
-// +kubebuilder:printcolumn:name="STATUS",type=string,JSONPath=`.status.phase`
-// +kubebuilder:printcolumn:name="COMPLETION-TIME",type=date,JSONPath=`.status.completionTimestamp`
-// +kubebuilder:printcolumn:name="AGE",type=date,JSONPath=`.metadata.creationTimestamp`
-
-// RestoreJob is the Schema for the restorejobs API (defined by User)
-type RestoreJob struct {
- metav1.TypeMeta `json:",inline"`
- metav1.ObjectMeta `json:"metadata,omitempty"`
-
- Spec RestoreJobSpec `json:"spec,omitempty"`
- Status RestoreJobStatus `json:"status,omitempty"`
-}
-
-// +kubebuilder:object:root=true
-
-// RestoreJobList contains a list of RestoreJob
-type RestoreJobList struct {
- metav1.TypeMeta `json:",inline"`
- metav1.ListMeta `json:"metadata,omitempty"`
- Items []RestoreJob `json:"items"`
-}
-
-func init() {
- SchemeBuilder.Register(&RestoreJob{}, &RestoreJobList{})
-}
diff --git a/apis/dataprotection/v1alpha1/types.go b/apis/dataprotection/v1alpha1/types.go
index 48e0c97c686..979f13c5e39 100644
--- a/apis/dataprotection/v1alpha1/types.go
+++ b/apis/dataprotection/v1alpha1/types.go
@@ -16,107 +16,209 @@ limitations under the License.
package v1alpha1
-// BackupPhase The current phase. Valid values are New, InProgress, Completed, Failed.
-// +enum
-// +kubebuilder:validation:Enum={New,InProgress,Running,Completed,Failed,Deleting}
-type BackupPhase string
-
-const (
- BackupNew BackupPhase = "New"
- BackupInProgress BackupPhase = "InProgress"
- BackupRunning BackupPhase = "Running"
- BackupCompleted BackupPhase = "Completed"
- BackupFailed BackupPhase = "Failed"
- BackupDeleting BackupPhase = "Deleting"
+import (
+ "errors"
+ "strconv"
+ "strings"
+ "time"
+ "unicode"
)
-// BackupType the backup type, marked backup set is datafile or logfile or snapshot.
+// Phase defines the BackupPolicy and ActionSet CR .status.phase
// +enum
-// +kubebuilder:validation:Enum={datafile,logfile,snapshot}
-type BackupType string
+// +kubebuilder:validation:Enum={Available,Unavailable}
+type Phase string
const (
- BackupTypeDataFile BackupType = "datafile"
- BackupTypeLogFile BackupType = "logfile"
- BackupTypeSnapshot BackupType = "snapshot"
+ AvailablePhase Phase = "Available"
+ UnavailablePhase Phase = "Unavailable"
)
-// BackupMethod the backup method
-// +enum
-// +kubebuilder:validation:Enum={snapshot,backupTool}
-type BackupMethod string
-
-const (
- BackupMethodSnapshot BackupMethod = "snapshot"
- BackupMethodBackupTool BackupMethod = "backupTool"
-)
+func (p Phase) IsAvailable() bool {
+ return p == AvailablePhase
+}
-// BaseBackupType the base backup type.
-// +enum
-// +kubebuilder:validation:Enum={full,snapshot}
-type BaseBackupType string
-
-// CreatePVCPolicy the policy how to create the PersistentVolumeClaim for backup.
-// +enum
-// +kubebuilder:validation:Enum={IfNotPresent,Never}
-type CreatePVCPolicy string
-
-const (
- CreatePVCPolicyNever CreatePVCPolicy = "Never"
- CreatePVCPolicyIfNotPresent CreatePVCPolicy = "IfNotPresent"
-)
-
-// BackupPolicyPhase defines phases for BackupPolicy CR.
+// BackupRepoPhase defines phases for BackupRepo CR.
// +enum
-// +kubebuilder:validation:Enum={Available,Failed}
-type BackupPolicyPhase string
+// +kubebuilder:validation:Enum={PreChecking,Failed,Ready,Deleting}
+type BackupRepoPhase string
const (
- PolicyAvailable BackupPolicyPhase = "Available"
- PolicyFailed BackupPolicyPhase = "Failed"
+ BackupRepoPreChecking BackupRepoPhase = "PreChecking"
+ BackupRepoFailed BackupRepoPhase = "Failed"
+ BackupRepoReady BackupRepoPhase = "Ready"
+ BackupRepoDeleting BackupRepoPhase = "Deleting"
)
-// RestoreJobPhase The current phase. Valid values are New, InProgressPhy, InProgressLogic, Completed, Failed.
+// RetentionPeriod represents a duration in the format "1y2mo3w4d5h6m", where
+// y=year, mo=month, w=week, d=day, h=hour, m=minute.
+type RetentionPeriod string
+
+// ToDuration converts the RetentionPeriod to time.Duration.
+func (r RetentionPeriod) ToDuration() (time.Duration, error) {
+ if len(r.String()) == 0 {
+ return time.Duration(0), nil
+ }
+
+ minutes, err := r.toMinutes()
+ if err != nil {
+ return time.Duration(0), err
+ }
+ return time.Minute * time.Duration(minutes), nil
+}
+
+func (r RetentionPeriod) String() string {
+ return string(r)
+}
+
+func (r RetentionPeriod) toMinutes() (int, error) {
+ d, err := r.parseDuration()
+ if err != nil {
+ return 0, err
+ }
+ minutes := d.Minutes
+ minutes += d.Hours * 60
+ minutes += d.Days * 24 * 60
+ minutes += d.Weeks * 7 * 24 * 60
+ minutes += d.Months * 30 * 24 * 60
+ minutes += d.Years * 365 * 24 * 60
+ return minutes, nil
+}
+
+type duration struct {
+ Minutes int
+ Hours int
+ Days int
+ Weeks int
+ Months int
+ Years int
+}
+
+var errInvalidDuration = errors.New("invalid duration provided")
+
+// parseDuration parses a duration from a string. The format is `6y5m234d37h`
+func (r RetentionPeriod) parseDuration() (duration, error) {
+ var (
+ d duration
+ num int
+ err error
+ )
+
+ s := strings.TrimSpace(r.String())
+ for s != "" {
+ num, s, err = r.nextNumber(s)
+ if err != nil {
+ return duration{}, err
+ }
+
+ if len(s) == 0 {
+ return duration{}, errInvalidDuration
+ }
+
+ if len(s) > 1 && s[0] == 'm' && s[1] == 'o' {
+ d.Months = num
+ s = s[2:]
+ continue
+ }
+
+ switch s[0] {
+ case 'y':
+ d.Years = num
+ case 'w':
+ d.Weeks = num
+ case 'd':
+ d.Days = num
+ case 'h':
+ d.Hours = num
+ case 'm':
+ d.Minutes = num
+ default:
+ return duration{}, errInvalidDuration
+ }
+ s = s[1:]
+ }
+ return d, nil
+}
+
+func (r RetentionPeriod) nextNumber(input string) (num int, rest string, err error) {
+ if len(input) == 0 {
+ return 0, "", nil
+ }
+
+ var (
+ n string
+ negative bool
+ )
+
+ if input[0] == '-' {
+ negative = true
+ input = input[1:]
+ }
+
+ for i, s := range input {
+ if !unicode.IsNumber(s) {
+ rest = input[i:]
+ break
+ }
+
+ n += string(s)
+ }
+
+ if len(n) == 0 {
+ return 0, input, errInvalidDuration
+ }
+
+ num, err = strconv.Atoi(n)
+ if err != nil {
+ return 0, input, err
+ }
+
+ if negative {
+ num = -num
+ }
+ return num, rest, nil
+}
+
+// RestorePhase The current phase. Valid values are Running, Completed, Failed, Deleting.
// +enum
-// +kubebuilder:validation:Enum={New,InProgressPhy,InProgressLogic,Completed,Failed}
-type RestoreJobPhase string
+// +kubebuilder:validation:Enum={Running,Completed,Failed,Deleting}
+type RestorePhase string
const (
- RestoreJobNew RestoreJobPhase = "New"
- RestoreJobInProgressPhy RestoreJobPhase = "InProgressPhy"
- RestoreJobInProgressLogic RestoreJobPhase = "InProgressLogic"
- RestoreJobCompleted RestoreJobPhase = "Completed"
- RestoreJobFailed RestoreJobPhase = "Failed"
+ RestorePhaseRunning RestorePhase = "Running"
+ RestorePhaseCompleted RestorePhase = "Completed"
+ RestorePhaseFailed RestorePhase = "Failed"
+ RestorePhaseDeleting RestorePhase = "Deleting"
)
-// DeployKind which kind for run a backup tool.
+// RestoreActionStatus the status of restore action.
// +enum
-// +kubebuilder:validation:Enum={job,statefulSet}
-type DeployKind string
+// +kubebuilder:validation:Enum={Processing,Completed,Failed}
+type RestoreActionStatus string
const (
- DeployKindJob DeployKind = "job"
- DeployKindStatefulSet DeployKind = "statefulSet"
+ RestoreActionProcessing RestoreActionStatus = "Processing"
+ RestoreActionCompleted RestoreActionStatus = "Completed"
+ RestoreActionFailed RestoreActionStatus = "Failed"
)
-// PodRestoreScope defines the scope pod for restore from backup.
-// +enum
-// +kubebuilder:validation:Enum={All,ReadWrite}
-type PodRestoreScope string
+type RestoreStage string
const (
- PodRestoreScopeAll = "All"
- PodRestoreScopeReadWrite = "ReadWrite"
+ PrepareData RestoreStage = "prepareData"
+ PostReady RestoreStage = "postReady"
)
-// BackupRepoPhase defines phases for BackupRepo CR.
+// VolumeClaimManagementPolicy defines recovery strategy for persistent volume claim.
+// Supported policies are as follows:
+// 1. Parallel: parallel recovery of persistent volume claim.
+// 2. Serial: restore the persistent volume claim in sequence, and wait until the
+// previous persistent volume claim is restored before restoring a new one.
// +enum
-// +kubebuilder:validation:Enum={PreChecking,Failed,Ready,Deleting}
-type BackupRepoPhase string
+// +kubebuilder:validation:Enum={Parallel,Serial}
+type VolumeClaimManagementPolicy string
const (
- BackupRepoPreChecking BackupRepoPhase = "PreChecking"
- BackupRepoFailed BackupRepoPhase = "Failed"
- BackupRepoReady BackupRepoPhase = "Ready"
- BackupRepoDeleting BackupRepoPhase = "Deleting"
+ ParallelManagementPolicy VolumeClaimManagementPolicy = "Parallel"
+ SerialManagementPolicy VolumeClaimManagementPolicy = "Serial"
)
diff --git a/apis/dataprotection/v1alpha1/zz_generated.deepcopy.go b/apis/dataprotection/v1alpha1/zz_generated.deepcopy.go
index 408d51001f1..19444298669 100644
--- a/apis/dataprotection/v1alpha1/zz_generated.deepcopy.go
+++ b/apis/dataprotection/v1alpha1/zz_generated.deepcopy.go
@@ -25,32 +25,32 @@ along with this program. If not, see .
package v1alpha1
import (
- corev1 "k8s.io/api/core/v1"
- "k8s.io/apimachinery/pkg/apis/meta/v1"
+ "k8s.io/api/core/v1"
+ metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
runtime "k8s.io/apimachinery/pkg/runtime"
)
// DeepCopyInto is an autogenerated deepcopy function, copying the receiver, writing into out. in must be non-nil.
-func (in *Backup) DeepCopyInto(out *Backup) {
+func (in *ActionSet) DeepCopyInto(out *ActionSet) {
*out = *in
out.TypeMeta = in.TypeMeta
in.ObjectMeta.DeepCopyInto(&out.ObjectMeta)
- out.Spec = in.Spec
- in.Status.DeepCopyInto(&out.Status)
+ in.Spec.DeepCopyInto(&out.Spec)
+ out.Status = in.Status
}
-// DeepCopy is an autogenerated deepcopy function, copying the receiver, creating a new Backup.
-func (in *Backup) DeepCopy() *Backup {
+// DeepCopy is an autogenerated deepcopy function, copying the receiver, creating a new ActionSet.
+func (in *ActionSet) DeepCopy() *ActionSet {
if in == nil {
return nil
}
- out := new(Backup)
+ out := new(ActionSet)
in.DeepCopyInto(out)
return out
}
// DeepCopyObject is an autogenerated deepcopy function, copying the receiver, creating a new runtime.Object.
-func (in *Backup) DeepCopyObject() runtime.Object {
+func (in *ActionSet) DeepCopyObject() runtime.Object {
if c := in.DeepCopy(); c != nil {
return c
}
@@ -58,31 +58,31 @@ func (in *Backup) DeepCopyObject() runtime.Object {
}
// DeepCopyInto is an autogenerated deepcopy function, copying the receiver, writing into out. in must be non-nil.
-func (in *BackupList) DeepCopyInto(out *BackupList) {
+func (in *ActionSetList) DeepCopyInto(out *ActionSetList) {
*out = *in
out.TypeMeta = in.TypeMeta
in.ListMeta.DeepCopyInto(&out.ListMeta)
if in.Items != nil {
in, out := &in.Items, &out.Items
- *out = make([]Backup, len(*in))
+ *out = make([]ActionSet, len(*in))
for i := range *in {
(*in)[i].DeepCopyInto(&(*out)[i])
}
}
}
-// DeepCopy is an autogenerated deepcopy function, copying the receiver, creating a new BackupList.
-func (in *BackupList) DeepCopy() *BackupList {
+// DeepCopy is an autogenerated deepcopy function, copying the receiver, creating a new ActionSetList.
+func (in *ActionSetList) DeepCopy() *ActionSetList {
if in == nil {
return nil
}
- out := new(BackupList)
+ out := new(ActionSetList)
in.DeepCopyInto(out)
return out
}
// DeepCopyObject is an autogenerated deepcopy function, copying the receiver, creating a new runtime.Object.
-func (in *BackupList) DeepCopyObject() runtime.Object {
+func (in *ActionSetList) DeepCopyObject() runtime.Object {
if c := in.DeepCopy(); c != nil {
return c
}
@@ -90,49 +90,148 @@ func (in *BackupList) DeepCopyObject() runtime.Object {
}
// DeepCopyInto is an autogenerated deepcopy function, copying the receiver, writing into out. in must be non-nil.
-func (in *BackupLogStatus) DeepCopyInto(out *BackupLogStatus) {
+func (in *ActionSetSpec) DeepCopyInto(out *ActionSetSpec) {
+ *out = *in
+ if in.Env != nil {
+ in, out := &in.Env, &out.Env
+ *out = make([]v1.EnvVar, len(*in))
+ for i := range *in {
+ (*in)[i].DeepCopyInto(&(*out)[i])
+ }
+ }
+ if in.EnvFrom != nil {
+ in, out := &in.EnvFrom, &out.EnvFrom
+ *out = make([]v1.EnvFromSource, len(*in))
+ for i := range *in {
+ (*in)[i].DeepCopyInto(&(*out)[i])
+ }
+ }
+ if in.Backup != nil {
+ in, out := &in.Backup, &out.Backup
+ *out = new(BackupActionSpec)
+ (*in).DeepCopyInto(*out)
+ }
+ if in.Restore != nil {
+ in, out := &in.Restore, &out.Restore
+ *out = new(RestoreActionSpec)
+ (*in).DeepCopyInto(*out)
+ }
+}
+
+// DeepCopy is an autogenerated deepcopy function, copying the receiver, creating a new ActionSetSpec.
+func (in *ActionSetSpec) DeepCopy() *ActionSetSpec {
+ if in == nil {
+ return nil
+ }
+ out := new(ActionSetSpec)
+ in.DeepCopyInto(out)
+ return out
+}
+
+// DeepCopyInto is an autogenerated deepcopy function, copying the receiver, writing into out. in must be non-nil.
+func (in *ActionSetStatus) DeepCopyInto(out *ActionSetStatus) {
+ *out = *in
+}
+
+// DeepCopy is an autogenerated deepcopy function, copying the receiver, creating a new ActionSetStatus.
+func (in *ActionSetStatus) DeepCopy() *ActionSetStatus {
+ if in == nil {
+ return nil
+ }
+ out := new(ActionSetStatus)
+ in.DeepCopyInto(out)
+ return out
+}
+
+// DeepCopyInto is an autogenerated deepcopy function, copying the receiver, writing into out. in must be non-nil.
+func (in *ActionSpec) DeepCopyInto(out *ActionSpec) {
+ *out = *in
+ if in.Exec != nil {
+ in, out := &in.Exec, &out.Exec
+ *out = new(ExecActionSpec)
+ (*in).DeepCopyInto(*out)
+ }
+ if in.Job != nil {
+ in, out := &in.Job, &out.Job
+ *out = new(JobActionSpec)
+ (*in).DeepCopyInto(*out)
+ }
+}
+
+// DeepCopy is an autogenerated deepcopy function, copying the receiver, creating a new ActionSpec.
+func (in *ActionSpec) DeepCopy() *ActionSpec {
+ if in == nil {
+ return nil
+ }
+ out := new(ActionSpec)
+ in.DeepCopyInto(out)
+ return out
+}
+
+// DeepCopyInto is an autogenerated deepcopy function, copying the receiver, writing into out. in must be non-nil.
+func (in *ActionStatus) DeepCopyInto(out *ActionStatus) {
*out = *in
- if in.StartTime != nil {
- in, out := &in.StartTime, &out.StartTime
+ if in.StartTimestamp != nil {
+ in, out := &in.StartTimestamp, &out.StartTimestamp
*out = (*in).DeepCopy()
}
- if in.StopTime != nil {
- in, out := &in.StopTime, &out.StopTime
+ if in.CompletionTimestamp != nil {
+ in, out := &in.CompletionTimestamp, &out.CompletionTimestamp
*out = (*in).DeepCopy()
}
+ if in.AvailableReplicas != nil {
+ in, out := &in.AvailableReplicas, &out.AvailableReplicas
+ *out = new(int32)
+ **out = **in
+ }
+ if in.ObjectRef != nil {
+ in, out := &in.ObjectRef, &out.ObjectRef
+ *out = new(v1.ObjectReference)
+ **out = **in
+ }
+ if in.TimeRange != nil {
+ in, out := &in.TimeRange, &out.TimeRange
+ *out = new(BackupTimeRange)
+ (*in).DeepCopyInto(*out)
+ }
+ if in.VolumeSnapshots != nil {
+ in, out := &in.VolumeSnapshots, &out.VolumeSnapshots
+ *out = make([]VolumeSnapshotStatus, len(*in))
+ copy(*out, *in)
+ }
}
-// DeepCopy is an autogenerated deepcopy function, copying the receiver, creating a new BackupLogStatus.
-func (in *BackupLogStatus) DeepCopy() *BackupLogStatus {
+// DeepCopy is an autogenerated deepcopy function, copying the receiver, creating a new ActionStatus.
+func (in *ActionStatus) DeepCopy() *ActionStatus {
if in == nil {
return nil
}
- out := new(BackupLogStatus)
+ out := new(ActionStatus)
in.DeepCopyInto(out)
return out
}
// DeepCopyInto is an autogenerated deepcopy function, copying the receiver, writing into out. in must be non-nil.
-func (in *BackupPolicy) DeepCopyInto(out *BackupPolicy) {
+func (in *Backup) DeepCopyInto(out *Backup) {
*out = *in
out.TypeMeta = in.TypeMeta
in.ObjectMeta.DeepCopyInto(&out.ObjectMeta)
- in.Spec.DeepCopyInto(&out.Spec)
+ out.Spec = in.Spec
in.Status.DeepCopyInto(&out.Status)
}
-// DeepCopy is an autogenerated deepcopy function, copying the receiver, creating a new BackupPolicy.
-func (in *BackupPolicy) DeepCopy() *BackupPolicy {
+// DeepCopy is an autogenerated deepcopy function, copying the receiver, creating a new Backup.
+func (in *Backup) DeepCopy() *Backup {
if in == nil {
return nil
}
- out := new(BackupPolicy)
+ out := new(Backup)
in.DeepCopyInto(out)
return out
}
// DeepCopyObject is an autogenerated deepcopy function, copying the receiver, creating a new runtime.Object.
-func (in *BackupPolicy) DeepCopyObject() runtime.Object {
+func (in *Backup) DeepCopyObject() runtime.Object {
if c := in.DeepCopy(); c != nil {
return c
}
@@ -140,56 +239,101 @@ func (in *BackupPolicy) DeepCopyObject() runtime.Object {
}
// DeepCopyInto is an autogenerated deepcopy function, copying the receiver, writing into out. in must be non-nil.
-func (in *BackupPolicyHook) DeepCopyInto(out *BackupPolicyHook) {
+func (in *BackupActionSpec) DeepCopyInto(out *BackupActionSpec) {
*out = *in
- if in.PreCommands != nil {
- in, out := &in.PreCommands, &out.PreCommands
- *out = make([]string, len(*in))
- copy(*out, *in)
+ if in.BackupData != nil {
+ in, out := &in.BackupData, &out.BackupData
+ *out = new(BackupDataActionSpec)
+ (*in).DeepCopyInto(*out)
}
- if in.PostCommands != nil {
- in, out := &in.PostCommands, &out.PostCommands
- *out = make([]string, len(*in))
- copy(*out, *in)
+ if in.PreBackup != nil {
+ in, out := &in.PreBackup, &out.PreBackup
+ *out = make([]ActionSpec, len(*in))
+ for i := range *in {
+ (*in)[i].DeepCopyInto(&(*out)[i])
+ }
+ }
+ if in.PostBackup != nil {
+ in, out := &in.PostBackup, &out.PostBackup
+ *out = make([]ActionSpec, len(*in))
+ for i := range *in {
+ (*in)[i].DeepCopyInto(&(*out)[i])
+ }
}
}
-// DeepCopy is an autogenerated deepcopy function, copying the receiver, creating a new BackupPolicyHook.
-func (in *BackupPolicyHook) DeepCopy() *BackupPolicyHook {
+// DeepCopy is an autogenerated deepcopy function, copying the receiver, creating a new BackupActionSpec.
+func (in *BackupActionSpec) DeepCopy() *BackupActionSpec {
if in == nil {
return nil
}
- out := new(BackupPolicyHook)
+ out := new(BackupActionSpec)
in.DeepCopyInto(out)
return out
}
// DeepCopyInto is an autogenerated deepcopy function, copying the receiver, writing into out. in must be non-nil.
-func (in *BackupPolicyList) DeepCopyInto(out *BackupPolicyList) {
+func (in *BackupConfig) DeepCopyInto(out *BackupConfig) {
+ *out = *in
+}
+
+// DeepCopy is an autogenerated deepcopy function, copying the receiver, creating a new BackupConfig.
+func (in *BackupConfig) DeepCopy() *BackupConfig {
+ if in == nil {
+ return nil
+ }
+ out := new(BackupConfig)
+ in.DeepCopyInto(out)
+ return out
+}
+
+// DeepCopyInto is an autogenerated deepcopy function, copying the receiver, writing into out. in must be non-nil.
+func (in *BackupDataActionSpec) DeepCopyInto(out *BackupDataActionSpec) {
+ *out = *in
+ in.JobActionSpec.DeepCopyInto(&out.JobActionSpec)
+ if in.SyncProgress != nil {
+ in, out := &in.SyncProgress, &out.SyncProgress
+ *out = new(SyncProgress)
+ (*in).DeepCopyInto(*out)
+ }
+}
+
+// DeepCopy is an autogenerated deepcopy function, copying the receiver, creating a new BackupDataActionSpec.
+func (in *BackupDataActionSpec) DeepCopy() *BackupDataActionSpec {
+ if in == nil {
+ return nil
+ }
+ out := new(BackupDataActionSpec)
+ in.DeepCopyInto(out)
+ return out
+}
+
+// DeepCopyInto is an autogenerated deepcopy function, copying the receiver, writing into out. in must be non-nil.
+func (in *BackupList) DeepCopyInto(out *BackupList) {
*out = *in
out.TypeMeta = in.TypeMeta
in.ListMeta.DeepCopyInto(&out.ListMeta)
if in.Items != nil {
in, out := &in.Items, &out.Items
- *out = make([]BackupPolicy, len(*in))
+ *out = make([]Backup, len(*in))
for i := range *in {
(*in)[i].DeepCopyInto(&(*out)[i])
}
}
}
-// DeepCopy is an autogenerated deepcopy function, copying the receiver, creating a new BackupPolicyList.
-func (in *BackupPolicyList) DeepCopy() *BackupPolicyList {
+// DeepCopy is an autogenerated deepcopy function, copying the receiver, creating a new BackupList.
+func (in *BackupList) DeepCopy() *BackupList {
if in == nil {
return nil
}
- out := new(BackupPolicyList)
+ out := new(BackupList)
in.DeepCopyInto(out)
return out
}
// DeepCopyObject is an autogenerated deepcopy function, copying the receiver, creating a new runtime.Object.
-func (in *BackupPolicyList) DeepCopyObject() runtime.Object {
+func (in *BackupList) DeepCopyObject() runtime.Object {
if c := in.DeepCopy(); c != nil {
return c
}
@@ -197,43 +341,125 @@ func (in *BackupPolicyList) DeepCopyObject() runtime.Object {
}
// DeepCopyInto is an autogenerated deepcopy function, copying the receiver, writing into out. in must be non-nil.
-func (in *BackupPolicySecret) DeepCopyInto(out *BackupPolicySecret) {
+func (in *BackupMethod) DeepCopyInto(out *BackupMethod) {
*out = *in
+ if in.SnapshotVolumes != nil {
+ in, out := &in.SnapshotVolumes, &out.SnapshotVolumes
+ *out = new(bool)
+ **out = **in
+ }
+ if in.TargetVolumes != nil {
+ in, out := &in.TargetVolumes, &out.TargetVolumes
+ *out = new(TargetVolumeInfo)
+ (*in).DeepCopyInto(*out)
+ }
+ if in.Env != nil {
+ in, out := &in.Env, &out.Env
+ *out = make([]v1.EnvVar, len(*in))
+ for i := range *in {
+ (*in)[i].DeepCopyInto(&(*out)[i])
+ }
+ }
+ if in.RuntimeSettings != nil {
+ in, out := &in.RuntimeSettings, &out.RuntimeSettings
+ *out = new(RuntimeSettings)
+ (*in).DeepCopyInto(*out)
+ }
}
-// DeepCopy is an autogenerated deepcopy function, copying the receiver, creating a new BackupPolicySecret.
-func (in *BackupPolicySecret) DeepCopy() *BackupPolicySecret {
+// DeepCopy is an autogenerated deepcopy function, copying the receiver, creating a new BackupMethod.
+func (in *BackupMethod) DeepCopy() *BackupMethod {
if in == nil {
return nil
}
- out := new(BackupPolicySecret)
+ out := new(BackupMethod)
in.DeepCopyInto(out)
return out
}
+// DeepCopyInto is an autogenerated deepcopy function, copying the receiver, writing into out. in must be non-nil.
+func (in *BackupPolicy) DeepCopyInto(out *BackupPolicy) {
+ *out = *in
+ out.TypeMeta = in.TypeMeta
+ in.ObjectMeta.DeepCopyInto(&out.ObjectMeta)
+ in.Spec.DeepCopyInto(&out.Spec)
+ out.Status = in.Status
+}
+
+// DeepCopy is an autogenerated deepcopy function, copying the receiver, creating a new BackupPolicy.
+func (in *BackupPolicy) DeepCopy() *BackupPolicy {
+ if in == nil {
+ return nil
+ }
+ out := new(BackupPolicy)
+ in.DeepCopyInto(out)
+ return out
+}
+
+// DeepCopyObject is an autogenerated deepcopy function, copying the receiver, creating a new runtime.Object.
+func (in *BackupPolicy) DeepCopyObject() runtime.Object {
+ if c := in.DeepCopy(); c != nil {
+ return c
+ }
+ return nil
+}
+
+// DeepCopyInto is an autogenerated deepcopy function, copying the receiver, writing into out. in must be non-nil.
+func (in *BackupPolicyList) DeepCopyInto(out *BackupPolicyList) {
+ *out = *in
+ out.TypeMeta = in.TypeMeta
+ in.ListMeta.DeepCopyInto(&out.ListMeta)
+ if in.Items != nil {
+ in, out := &in.Items, &out.Items
+ *out = make([]BackupPolicy, len(*in))
+ for i := range *in {
+ (*in)[i].DeepCopyInto(&(*out)[i])
+ }
+ }
+}
+
+// DeepCopy is an autogenerated deepcopy function, copying the receiver, creating a new BackupPolicyList.
+func (in *BackupPolicyList) DeepCopy() *BackupPolicyList {
+ if in == nil {
+ return nil
+ }
+ out := new(BackupPolicyList)
+ in.DeepCopyInto(out)
+ return out
+}
+
+// DeepCopyObject is an autogenerated deepcopy function, copying the receiver, creating a new runtime.Object.
+func (in *BackupPolicyList) DeepCopyObject() runtime.Object {
+ if c := in.DeepCopy(); c != nil {
+ return c
+ }
+ return nil
+}
+
// DeepCopyInto is an autogenerated deepcopy function, copying the receiver, writing into out. in must be non-nil.
func (in *BackupPolicySpec) DeepCopyInto(out *BackupPolicySpec) {
*out = *in
- if in.Retention != nil {
- in, out := &in.Retention, &out.Retention
- *out = new(RetentionSpec)
- (*in).DeepCopyInto(*out)
+ if in.BackupRepoName != nil {
+ in, out := &in.BackupRepoName, &out.BackupRepoName
+ *out = new(string)
+ **out = **in
}
- in.Schedule.DeepCopyInto(&out.Schedule)
- if in.Snapshot != nil {
- in, out := &in.Snapshot, &out.Snapshot
- *out = new(SnapshotPolicy)
- (*in).DeepCopyInto(*out)
+ if in.BackoffLimit != nil {
+ in, out := &in.BackoffLimit, &out.BackoffLimit
+ *out = new(int32)
+ **out = **in
}
- if in.Datafile != nil {
- in, out := &in.Datafile, &out.Datafile
- *out = new(CommonBackupPolicy)
+ if in.Target != nil {
+ in, out := &in.Target, &out.Target
+ *out = new(BackupTarget)
(*in).DeepCopyInto(*out)
}
- if in.Logfile != nil {
- in, out := &in.Logfile, &out.Logfile
- *out = new(CommonBackupPolicy)
- (*in).DeepCopyInto(*out)
+ if in.BackupMethods != nil {
+ in, out := &in.BackupMethods, &out.BackupMethods
+ *out = make([]BackupMethod, len(*in))
+ for i := range *in {
+ (*in)[i].DeepCopyInto(&(*out)[i])
+ }
}
}
@@ -250,14 +476,6 @@ func (in *BackupPolicySpec) DeepCopy() *BackupPolicySpec {
// DeepCopyInto is an autogenerated deepcopy function, copying the receiver, writing into out. in must be non-nil.
func (in *BackupPolicyStatus) DeepCopyInto(out *BackupPolicyStatus) {
*out = *in
- if in.LastScheduleTime != nil {
- in, out := &in.LastScheduleTime, &out.LastScheduleTime
- *out = (*in).DeepCopy()
- }
- if in.LastSuccessfulTime != nil {
- in, out := &in.LastSuccessfulTime, &out.LastSuccessfulTime
- *out = (*in).DeepCopy()
- }
}
// DeepCopy is an autogenerated deepcopy function, copying the receiver, creating a new BackupPolicyStatus.
@@ -342,7 +560,7 @@ func (in *BackupRepoSpec) DeepCopyInto(out *BackupRepoSpec) {
}
if in.Credential != nil {
in, out := &in.Credential, &out.Credential
- *out = new(corev1.SecretReference)
+ *out = new(v1.SecretReference)
**out = **in
}
}
@@ -362,14 +580,14 @@ func (in *BackupRepoStatus) DeepCopyInto(out *BackupRepoStatus) {
*out = *in
if in.Conditions != nil {
in, out := &in.Conditions, &out.Conditions
- *out = make([]v1.Condition, len(*in))
+ *out = make([]metav1.Condition, len(*in))
for i := range *in {
(*in)[i].DeepCopyInto(&(*out)[i])
}
}
if in.GeneratedCSIDriverSecret != nil {
in, out := &in.GeneratedCSIDriverSecret, &out.GeneratedCSIDriverSecret
- *out = new(corev1.SecretReference)
+ *out = new(v1.SecretReference)
**out = **in
}
}
@@ -385,16 +603,109 @@ func (in *BackupRepoStatus) DeepCopy() *BackupRepoStatus {
}
// DeepCopyInto is an autogenerated deepcopy function, copying the receiver, writing into out. in must be non-nil.
-func (in *BackupSnapshotStatus) DeepCopyInto(out *BackupSnapshotStatus) {
+func (in *BackupSchedule) DeepCopyInto(out *BackupSchedule) {
+ *out = *in
+ out.TypeMeta = in.TypeMeta
+ in.ObjectMeta.DeepCopyInto(&out.ObjectMeta)
+ in.Spec.DeepCopyInto(&out.Spec)
+ in.Status.DeepCopyInto(&out.Status)
+}
+
+// DeepCopy is an autogenerated deepcopy function, copying the receiver, creating a new BackupSchedule.
+func (in *BackupSchedule) DeepCopy() *BackupSchedule {
+ if in == nil {
+ return nil
+ }
+ out := new(BackupSchedule)
+ in.DeepCopyInto(out)
+ return out
+}
+
+// DeepCopyObject is an autogenerated deepcopy function, copying the receiver, creating a new runtime.Object.
+func (in *BackupSchedule) DeepCopyObject() runtime.Object {
+ if c := in.DeepCopy(); c != nil {
+ return c
+ }
+ return nil
+}
+
+// DeepCopyInto is an autogenerated deepcopy function, copying the receiver, writing into out. in must be non-nil.
+func (in *BackupScheduleList) DeepCopyInto(out *BackupScheduleList) {
*out = *in
+ out.TypeMeta = in.TypeMeta
+ in.ListMeta.DeepCopyInto(&out.ListMeta)
+ if in.Items != nil {
+ in, out := &in.Items, &out.Items
+ *out = make([]BackupSchedule, len(*in))
+ for i := range *in {
+ (*in)[i].DeepCopyInto(&(*out)[i])
+ }
+ }
}
-// DeepCopy is an autogenerated deepcopy function, copying the receiver, creating a new BackupSnapshotStatus.
-func (in *BackupSnapshotStatus) DeepCopy() *BackupSnapshotStatus {
+// DeepCopy is an autogenerated deepcopy function, copying the receiver, creating a new BackupScheduleList.
+func (in *BackupScheduleList) DeepCopy() *BackupScheduleList {
if in == nil {
return nil
}
- out := new(BackupSnapshotStatus)
+ out := new(BackupScheduleList)
+ in.DeepCopyInto(out)
+ return out
+}
+
+// DeepCopyObject is an autogenerated deepcopy function, copying the receiver, creating a new runtime.Object.
+func (in *BackupScheduleList) DeepCopyObject() runtime.Object {
+ if c := in.DeepCopy(); c != nil {
+ return c
+ }
+ return nil
+}
+
+// DeepCopyInto is an autogenerated deepcopy function, copying the receiver, writing into out. in must be non-nil.
+func (in *BackupScheduleSpec) DeepCopyInto(out *BackupScheduleSpec) {
+ *out = *in
+ if in.StartingDeadlineMinutes != nil {
+ in, out := &in.StartingDeadlineMinutes, &out.StartingDeadlineMinutes
+ *out = new(int64)
+ **out = **in
+ }
+ if in.Schedules != nil {
+ in, out := &in.Schedules, &out.Schedules
+ *out = make([]SchedulePolicy, len(*in))
+ for i := range *in {
+ (*in)[i].DeepCopyInto(&(*out)[i])
+ }
+ }
+}
+
+// DeepCopy is an autogenerated deepcopy function, copying the receiver, creating a new BackupScheduleSpec.
+func (in *BackupScheduleSpec) DeepCopy() *BackupScheduleSpec {
+ if in == nil {
+ return nil
+ }
+ out := new(BackupScheduleSpec)
+ in.DeepCopyInto(out)
+ return out
+}
+
+// DeepCopyInto is an autogenerated deepcopy function, copying the receiver, writing into out. in must be non-nil.
+func (in *BackupScheduleStatus) DeepCopyInto(out *BackupScheduleStatus) {
+ *out = *in
+ if in.Schedules != nil {
+ in, out := &in.Schedules, &out.Schedules
+ *out = make(map[string]ScheduleStatus, len(*in))
+ for key, val := range *in {
+ (*out)[key] = *val.DeepCopy()
+ }
+ }
+}
+
+// DeepCopy is an autogenerated deepcopy function, copying the receiver, creating a new BackupScheduleStatus.
+func (in *BackupScheduleStatus) DeepCopy() *BackupScheduleStatus {
+ if in == nil {
+ return nil
+ }
+ out := new(BackupScheduleStatus)
in.DeepCopyInto(out)
return out
}
@@ -431,600 +742,865 @@ func (in *BackupStatus) DeepCopyInto(out *BackupStatus) {
}
if in.Duration != nil {
in, out := &in.Duration, &out.Duration
- *out = new(v1.Duration)
+ *out = new(metav1.Duration)
**out = **in
}
- if in.AvailableReplicas != nil {
- in, out := &in.AvailableReplicas, &out.AvailableReplicas
- *out = new(int32)
+ if in.TimeRange != nil {
+ in, out := &in.TimeRange, &out.TimeRange
+ *out = new(BackupTimeRange)
+ (*in).DeepCopyInto(*out)
+ }
+ if in.Target != nil {
+ in, out := &in.Target, &out.Target
+ *out = new(BackupTarget)
+ (*in).DeepCopyInto(*out)
+ }
+ if in.BackupMethod != nil {
+ in, out := &in.BackupMethod, &out.BackupMethod
+ *out = new(BackupMethod)
+ (*in).DeepCopyInto(*out)
+ }
+ if in.Actions != nil {
+ in, out := &in.Actions, &out.Actions
+ *out = make([]ActionStatus, len(*in))
+ for i := range *in {
+ (*in)[i].DeepCopyInto(&(*out)[i])
+ }
+ }
+ if in.VolumeSnapshots != nil {
+ in, out := &in.VolumeSnapshots, &out.VolumeSnapshots
+ *out = make([]VolumeSnapshotStatus, len(*in))
+ copy(*out, *in)
+ }
+}
+
+// DeepCopy is an autogenerated deepcopy function, copying the receiver, creating a new BackupStatus.
+func (in *BackupStatus) DeepCopy() *BackupStatus {
+ if in == nil {
+ return nil
+ }
+ out := new(BackupStatus)
+ in.DeepCopyInto(out)
+ return out
+}
+
+// DeepCopyInto is an autogenerated deepcopy function, copying the receiver, writing into out. in must be non-nil.
+func (in *BackupTarget) DeepCopyInto(out *BackupTarget) {
+ *out = *in
+ if in.PodSelector != nil {
+ in, out := &in.PodSelector, &out.PodSelector
+ *out = new(PodSelector)
+ (*in).DeepCopyInto(*out)
+ }
+ if in.ConnectionCredential != nil {
+ in, out := &in.ConnectionCredential, &out.ConnectionCredential
+ *out = new(ConnectionCredential)
+ **out = **in
+ }
+ if in.Resources != nil {
+ in, out := &in.Resources, &out.Resources
+ *out = new(KubeResources)
+ (*in).DeepCopyInto(*out)
+ }
+}
+
+// DeepCopy is an autogenerated deepcopy function, copying the receiver, creating a new BackupTarget.
+func (in *BackupTarget) DeepCopy() *BackupTarget {
+ if in == nil {
+ return nil
+ }
+ out := new(BackupTarget)
+ in.DeepCopyInto(out)
+ return out
+}
+
+// DeepCopyInto is an autogenerated deepcopy function, copying the receiver, writing into out. in must be non-nil.
+func (in *BackupTimeRange) DeepCopyInto(out *BackupTimeRange) {
+ *out = *in
+ if in.Start != nil {
+ in, out := &in.Start, &out.Start
+ *out = (*in).DeepCopy()
+ }
+ if in.End != nil {
+ in, out := &in.End, &out.End
+ *out = (*in).DeepCopy()
+ }
+}
+
+// DeepCopy is an autogenerated deepcopy function, copying the receiver, creating a new BackupTimeRange.
+func (in *BackupTimeRange) DeepCopy() *BackupTimeRange {
+ if in == nil {
+ return nil
+ }
+ out := new(BackupTimeRange)
+ in.DeepCopyInto(out)
+ return out
+}
+
+// DeepCopyInto is an autogenerated deepcopy function, copying the receiver, writing into out. in must be non-nil.
+func (in *ConnectCredential) DeepCopyInto(out *ConnectCredential) {
+ *out = *in
+}
+
+// DeepCopy is an autogenerated deepcopy function, copying the receiver, creating a new ConnectCredential.
+func (in *ConnectCredential) DeepCopy() *ConnectCredential {
+ if in == nil {
+ return nil
+ }
+ out := new(ConnectCredential)
+ in.DeepCopyInto(out)
+ return out
+}
+
+// DeepCopyInto is an autogenerated deepcopy function, copying the receiver, writing into out. in must be non-nil.
+func (in *ConnectionCredential) DeepCopyInto(out *ConnectionCredential) {
+ *out = *in
+}
+
+// DeepCopy is an autogenerated deepcopy function, copying the receiver, creating a new ConnectionCredential.
+func (in *ConnectionCredential) DeepCopy() *ConnectionCredential {
+ if in == nil {
+ return nil
+ }
+ out := new(ConnectionCredential)
+ in.DeepCopyInto(out)
+ return out
+}
+
+// DeepCopyInto is an autogenerated deepcopy function, copying the receiver, writing into out. in must be non-nil.
+func (in *ExecAction) DeepCopyInto(out *ExecAction) {
+ *out = *in
+ in.Target.DeepCopyInto(&out.Target)
+}
+
+// DeepCopy is an autogenerated deepcopy function, copying the receiver, creating a new ExecAction.
+func (in *ExecAction) DeepCopy() *ExecAction {
+ if in == nil {
+ return nil
+ }
+ out := new(ExecAction)
+ in.DeepCopyInto(out)
+ return out
+}
+
+// DeepCopyInto is an autogenerated deepcopy function, copying the receiver, writing into out. in must be non-nil.
+func (in *ExecActionSpec) DeepCopyInto(out *ExecActionSpec) {
+ *out = *in
+ if in.Command != nil {
+ in, out := &in.Command, &out.Command
+ *out = make([]string, len(*in))
+ copy(*out, *in)
+ }
+ out.Timeout = in.Timeout
+}
+
+// DeepCopy is an autogenerated deepcopy function, copying the receiver, creating a new ExecActionSpec.
+func (in *ExecActionSpec) DeepCopy() *ExecActionSpec {
+ if in == nil {
+ return nil
+ }
+ out := new(ExecActionSpec)
+ in.DeepCopyInto(out)
+ return out
+}
+
+// DeepCopyInto is an autogenerated deepcopy function, copying the receiver, writing into out. in must be non-nil.
+func (in *ExecActionTarget) DeepCopyInto(out *ExecActionTarget) {
+ *out = *in
+ in.PodSelector.DeepCopyInto(&out.PodSelector)
+}
+
+// DeepCopy is an autogenerated deepcopy function, copying the receiver, creating a new ExecActionTarget.
+func (in *ExecActionTarget) DeepCopy() *ExecActionTarget {
+ if in == nil {
+ return nil
+ }
+ out := new(ExecActionTarget)
+ in.DeepCopyInto(out)
+ return out
+}
+
+// DeepCopyInto is an autogenerated deepcopy function, copying the receiver, writing into out. in must be non-nil.
+func (in *IncludeResource) DeepCopyInto(out *IncludeResource) {
+ *out = *in
+ in.LabelSelector.DeepCopyInto(&out.LabelSelector)
+}
+
+// DeepCopy is an autogenerated deepcopy function, copying the receiver, creating a new IncludeResource.
+func (in *IncludeResource) DeepCopy() *IncludeResource {
+ if in == nil {
+ return nil
+ }
+ out := new(IncludeResource)
+ in.DeepCopyInto(out)
+ return out
+}
+
+// DeepCopyInto is an autogenerated deepcopy function, copying the receiver, writing into out. in must be non-nil.
+func (in *JobAction) DeepCopyInto(out *JobAction) {
+ *out = *in
+ in.Target.DeepCopyInto(&out.Target)
+}
+
+// DeepCopy is an autogenerated deepcopy function, copying the receiver, creating a new JobAction.
+func (in *JobAction) DeepCopy() *JobAction {
+ if in == nil {
+ return nil
+ }
+ out := new(JobAction)
+ in.DeepCopyInto(out)
+ return out
+}
+
+// DeepCopyInto is an autogenerated deepcopy function, copying the receiver, writing into out. in must be non-nil.
+func (in *JobActionSpec) DeepCopyInto(out *JobActionSpec) {
+ *out = *in
+ if in.RunOnTargetPodNode != nil {
+ in, out := &in.RunOnTargetPodNode, &out.RunOnTargetPodNode
+ *out = new(bool)
**out = **in
}
- if in.Manifests != nil {
- in, out := &in.Manifests, &out.Manifests
- *out = new(ManifestsStatus)
- (*in).DeepCopyInto(*out)
+ if in.Command != nil {
+ in, out := &in.Command, &out.Command
+ *out = make([]string, len(*in))
+ copy(*out, *in)
}
}
-// DeepCopy is an autogenerated deepcopy function, copying the receiver, creating a new BackupStatus.
-func (in *BackupStatus) DeepCopy() *BackupStatus {
+// DeepCopy is an autogenerated deepcopy function, copying the receiver, creating a new JobActionSpec.
+func (in *JobActionSpec) DeepCopy() *JobActionSpec {
if in == nil {
return nil
}
- out := new(BackupStatus)
+ out := new(JobActionSpec)
in.DeepCopyInto(out)
return out
}
// DeepCopyInto is an autogenerated deepcopy function, copying the receiver, writing into out. in must be non-nil.
-func (in *BackupStatusUpdate) DeepCopyInto(out *BackupStatusUpdate) {
+func (in *JobActionTarget) DeepCopyInto(out *JobActionTarget) {
*out = *in
+ in.PodSelector.DeepCopyInto(&out.PodSelector)
+ if in.VolumeMounts != nil {
+ in, out := &in.VolumeMounts, &out.VolumeMounts
+ *out = make([]v1.VolumeMount, len(*in))
+ for i := range *in {
+ (*in)[i].DeepCopyInto(&(*out)[i])
+ }
+ }
}
-// DeepCopy is an autogenerated deepcopy function, copying the receiver, creating a new BackupStatusUpdate.
-func (in *BackupStatusUpdate) DeepCopy() *BackupStatusUpdate {
+// DeepCopy is an autogenerated deepcopy function, copying the receiver, creating a new JobActionTarget.
+func (in *JobActionTarget) DeepCopy() *JobActionTarget {
if in == nil {
return nil
}
- out := new(BackupStatusUpdate)
+ out := new(JobActionTarget)
in.DeepCopyInto(out)
return out
}
// DeepCopyInto is an autogenerated deepcopy function, copying the receiver, writing into out. in must be non-nil.
-func (in *BackupTool) DeepCopyInto(out *BackupTool) {
+func (in *KubeResources) DeepCopyInto(out *KubeResources) {
*out = *in
- out.TypeMeta = in.TypeMeta
- in.ObjectMeta.DeepCopyInto(&out.ObjectMeta)
- in.Spec.DeepCopyInto(&out.Spec)
- out.Status = in.Status
+ if in.Selector != nil {
+ in, out := &in.Selector, &out.Selector
+ *out = new(metav1.LabelSelector)
+ (*in).DeepCopyInto(*out)
+ }
+ if in.Included != nil {
+ in, out := &in.Included, &out.Included
+ *out = make([]string, len(*in))
+ copy(*out, *in)
+ }
+ if in.Excluded != nil {
+ in, out := &in.Excluded, &out.Excluded
+ *out = make([]string, len(*in))
+ copy(*out, *in)
+ }
}
-// DeepCopy is an autogenerated deepcopy function, copying the receiver, creating a new BackupTool.
-func (in *BackupTool) DeepCopy() *BackupTool {
+// DeepCopy is an autogenerated deepcopy function, copying the receiver, creating a new KubeResources.
+func (in *KubeResources) DeepCopy() *KubeResources {
if in == nil {
return nil
}
- out := new(BackupTool)
+ out := new(KubeResources)
in.DeepCopyInto(out)
return out
}
-// DeepCopyObject is an autogenerated deepcopy function, copying the receiver, creating a new runtime.Object.
-func (in *BackupTool) DeepCopyObject() runtime.Object {
- if c := in.DeepCopy(); c != nil {
- return c
+// DeepCopyInto is an autogenerated deepcopy function, copying the receiver, writing into out. in must be non-nil.
+func (in *PodSelector) DeepCopyInto(out *PodSelector) {
+ *out = *in
+ if in.LabelSelector != nil {
+ in, out := &in.LabelSelector, &out.LabelSelector
+ *out = new(metav1.LabelSelector)
+ (*in).DeepCopyInto(*out)
}
- return nil
+}
+
+// DeepCopy is an autogenerated deepcopy function, copying the receiver, creating a new PodSelector.
+func (in *PodSelector) DeepCopy() *PodSelector {
+ if in == nil {
+ return nil
+ }
+ out := new(PodSelector)
+ in.DeepCopyInto(out)
+ return out
}
// DeepCopyInto is an autogenerated deepcopy function, copying the receiver, writing into out. in must be non-nil.
-func (in *BackupToolList) DeepCopyInto(out *BackupToolList) {
+func (in *PrepareDataConfig) DeepCopyInto(out *PrepareDataConfig) {
*out = *in
- out.TypeMeta = in.TypeMeta
- in.ListMeta.DeepCopyInto(&out.ListMeta)
- if in.Items != nil {
- in, out := &in.Items, &out.Items
- *out = make([]BackupTool, len(*in))
+ if in.DataSourceRef != nil {
+ in, out := &in.DataSourceRef, &out.DataSourceRef
+ *out = new(VolumeConfig)
+ **out = **in
+ }
+ if in.RestoreVolumeClaims != nil {
+ in, out := &in.RestoreVolumeClaims, &out.RestoreVolumeClaims
+ *out = make([]RestoreVolumeClaim, len(*in))
for i := range *in {
(*in)[i].DeepCopyInto(&(*out)[i])
}
}
+ if in.RestoreVolumeClaimsTemplate != nil {
+ in, out := &in.RestoreVolumeClaimsTemplate, &out.RestoreVolumeClaimsTemplate
+ *out = new(RestoreVolumeClaimsTemplate)
+ (*in).DeepCopyInto(*out)
+ }
+ in.SchedulingSpec.DeepCopyInto(&out.SchedulingSpec)
}
-// DeepCopy is an autogenerated deepcopy function, copying the receiver, creating a new BackupToolList.
-func (in *BackupToolList) DeepCopy() *BackupToolList {
+// DeepCopy is an autogenerated deepcopy function, copying the receiver, creating a new PrepareDataConfig.
+func (in *PrepareDataConfig) DeepCopy() *PrepareDataConfig {
if in == nil {
return nil
}
- out := new(BackupToolList)
+ out := new(PrepareDataConfig)
in.DeepCopyInto(out)
return out
}
-// DeepCopyObject is an autogenerated deepcopy function, copying the receiver, creating a new runtime.Object.
-func (in *BackupToolList) DeepCopyObject() runtime.Object {
- if c := in.DeepCopy(); c != nil {
- return c
- }
- return nil
-}
-
// DeepCopyInto is an autogenerated deepcopy function, copying the receiver, writing into out. in must be non-nil.
-func (in *BackupToolManifestsStatus) DeepCopyInto(out *BackupToolManifestsStatus) {
+func (in *ReadinessProbe) DeepCopyInto(out *ReadinessProbe) {
*out = *in
+ in.Exec.DeepCopyInto(&out.Exec)
}
-// DeepCopy is an autogenerated deepcopy function, copying the receiver, creating a new BackupToolManifestsStatus.
-func (in *BackupToolManifestsStatus) DeepCopy() *BackupToolManifestsStatus {
+// DeepCopy is an autogenerated deepcopy function, copying the receiver, creating a new ReadinessProbe.
+func (in *ReadinessProbe) DeepCopy() *ReadinessProbe {
if in == nil {
return nil
}
- out := new(BackupToolManifestsStatus)
+ out := new(ReadinessProbe)
in.DeepCopyInto(out)
return out
}
// DeepCopyInto is an autogenerated deepcopy function, copying the receiver, writing into out. in must be non-nil.
-func (in *BackupToolRestoreCommand) DeepCopyInto(out *BackupToolRestoreCommand) {
+func (in *ReadinessProbeExecAction) DeepCopyInto(out *ReadinessProbeExecAction) {
*out = *in
- if in.RestoreCommands != nil {
- in, out := &in.RestoreCommands, &out.RestoreCommands
- *out = make([]string, len(*in))
- copy(*out, *in)
- }
- if in.IncrementalRestoreCommands != nil {
- in, out := &in.IncrementalRestoreCommands, &out.IncrementalRestoreCommands
+ if in.Command != nil {
+ in, out := &in.Command, &out.Command
*out = make([]string, len(*in))
copy(*out, *in)
}
}
-// DeepCopy is an autogenerated deepcopy function, copying the receiver, creating a new BackupToolRestoreCommand.
-func (in *BackupToolRestoreCommand) DeepCopy() *BackupToolRestoreCommand {
+// DeepCopy is an autogenerated deepcopy function, copying the receiver, creating a new ReadinessProbeExecAction.
+func (in *ReadinessProbeExecAction) DeepCopy() *ReadinessProbeExecAction {
if in == nil {
return nil
}
- out := new(BackupToolRestoreCommand)
+ out := new(ReadinessProbeExecAction)
in.DeepCopyInto(out)
return out
}
// DeepCopyInto is an autogenerated deepcopy function, copying the receiver, writing into out. in must be non-nil.
-func (in *BackupToolSpec) DeepCopyInto(out *BackupToolSpec) {
+func (in *ReadyConfig) DeepCopyInto(out *ReadyConfig) {
*out = *in
- if in.Resources != nil {
- in, out := &in.Resources, &out.Resources
- *out = new(corev1.ResourceRequirements)
+ if in.JobAction != nil {
+ in, out := &in.JobAction, &out.JobAction
+ *out = new(JobAction)
(*in).DeepCopyInto(*out)
}
- if in.Env != nil {
- in, out := &in.Env, &out.Env
- *out = make([]corev1.EnvVar, len(*in))
- for i := range *in {
- (*in)[i].DeepCopyInto(&(*out)[i])
- }
- }
- if in.EnvFrom != nil {
- in, out := &in.EnvFrom, &out.EnvFrom
- *out = make([]corev1.EnvFromSource, len(*in))
- for i := range *in {
- (*in)[i].DeepCopyInto(&(*out)[i])
- }
- }
- if in.BackupCommands != nil {
- in, out := &in.BackupCommands, &out.BackupCommands
- *out = make([]string, len(*in))
- copy(*out, *in)
- }
- if in.IncrementalBackupCommands != nil {
- in, out := &in.IncrementalBackupCommands, &out.IncrementalBackupCommands
- *out = make([]string, len(*in))
- copy(*out, *in)
- }
- if in.Physical != nil {
- in, out := &in.Physical, &out.Physical
- *out = new(PhysicalConfig)
+ if in.ExecAction != nil {
+ in, out := &in.ExecAction, &out.ExecAction
+ *out = new(ExecAction)
(*in).DeepCopyInto(*out)
}
- if in.Logical != nil {
- in, out := &in.Logical, &out.Logical
- *out = new(LogicalConfig)
+ if in.ConnectCredential != nil {
+ in, out := &in.ConnectCredential, &out.ConnectCredential
+ *out = new(ConnectCredential)
+ **out = **in
+ }
+ if in.ReadinessProbe != nil {
+ in, out := &in.ReadinessProbe, &out.ReadinessProbe
+ *out = new(ReadinessProbe)
(*in).DeepCopyInto(*out)
}
}
-// DeepCopy is an autogenerated deepcopy function, copying the receiver, creating a new BackupToolSpec.
-func (in *BackupToolSpec) DeepCopy() *BackupToolSpec {
+// DeepCopy is an autogenerated deepcopy function, copying the receiver, creating a new ReadyConfig.
+func (in *ReadyConfig) DeepCopy() *ReadyConfig {
if in == nil {
return nil
}
- out := new(BackupToolSpec)
+ out := new(ReadyConfig)
in.DeepCopyInto(out)
return out
}
// DeepCopyInto is an autogenerated deepcopy function, copying the receiver, writing into out. in must be non-nil.
-func (in *BackupToolStatus) DeepCopyInto(out *BackupToolStatus) {
+func (in *Restore) DeepCopyInto(out *Restore) {
*out = *in
+ out.TypeMeta = in.TypeMeta
+ in.ObjectMeta.DeepCopyInto(&out.ObjectMeta)
+ in.Spec.DeepCopyInto(&out.Spec)
+ in.Status.DeepCopyInto(&out.Status)
}
-// DeepCopy is an autogenerated deepcopy function, copying the receiver, creating a new BackupToolStatus.
-func (in *BackupToolStatus) DeepCopy() *BackupToolStatus {
+// DeepCopy is an autogenerated deepcopy function, copying the receiver, creating a new Restore.
+func (in *Restore) DeepCopy() *Restore {
if in == nil {
return nil
}
- out := new(BackupToolStatus)
+ out := new(Restore)
in.DeepCopyInto(out)
return out
}
+// DeepCopyObject is an autogenerated deepcopy function, copying the receiver, creating a new runtime.Object.
+func (in *Restore) DeepCopyObject() runtime.Object {
+ if c := in.DeepCopy(); c != nil {
+ return c
+ }
+ return nil
+}
+
// DeepCopyInto is an autogenerated deepcopy function, copying the receiver, writing into out. in must be non-nil.
-func (in *BasePolicy) DeepCopyInto(out *BasePolicy) {
+func (in *RestoreActionSpec) DeepCopyInto(out *RestoreActionSpec) {
*out = *in
- in.Target.DeepCopyInto(&out.Target)
- if in.BackupStatusUpdates != nil {
- in, out := &in.BackupStatusUpdates, &out.BackupStatusUpdates
- *out = make([]BackupStatusUpdate, len(*in))
- copy(*out, *in)
+ if in.PrepareData != nil {
+ in, out := &in.PrepareData, &out.PrepareData
+ *out = new(JobActionSpec)
+ (*in).DeepCopyInto(*out)
+ }
+ if in.PostReady != nil {
+ in, out := &in.PostReady, &out.PostReady
+ *out = make([]ActionSpec, len(*in))
+ for i := range *in {
+ (*in)[i].DeepCopyInto(&(*out)[i])
+ }
}
}
-// DeepCopy is an autogenerated deepcopy function, copying the receiver, creating a new BasePolicy.
-func (in *BasePolicy) DeepCopy() *BasePolicy {
+// DeepCopy is an autogenerated deepcopy function, copying the receiver, creating a new RestoreActionSpec.
+func (in *RestoreActionSpec) DeepCopy() *RestoreActionSpec {
if in == nil {
return nil
}
- out := new(BasePolicy)
+ out := new(RestoreActionSpec)
in.DeepCopyInto(out)
return out
}
// DeepCopyInto is an autogenerated deepcopy function, copying the receiver, writing into out. in must be non-nil.
-func (in *CommonBackupPolicy) DeepCopyInto(out *CommonBackupPolicy) {
+func (in *RestoreKubeResources) DeepCopyInto(out *RestoreKubeResources) {
*out = *in
- in.BasePolicy.DeepCopyInto(&out.BasePolicy)
- in.PersistentVolumeClaim.DeepCopyInto(&out.PersistentVolumeClaim)
- if in.BackupRepoName != nil {
- in, out := &in.BackupRepoName, &out.BackupRepoName
- *out = new(string)
- **out = **in
+ if in.IncludeResources != nil {
+ in, out := &in.IncludeResources, &out.IncludeResources
+ *out = make([]IncludeResource, len(*in))
+ for i := range *in {
+ (*in)[i].DeepCopyInto(&(*out)[i])
+ }
}
}
-// DeepCopy is an autogenerated deepcopy function, copying the receiver, creating a new CommonBackupPolicy.
-func (in *CommonBackupPolicy) DeepCopy() *CommonBackupPolicy {
+// DeepCopy is an autogenerated deepcopy function, copying the receiver, creating a new RestoreKubeResources.
+func (in *RestoreKubeResources) DeepCopy() *RestoreKubeResources {
if in == nil {
return nil
}
- out := new(CommonBackupPolicy)
+ out := new(RestoreKubeResources)
in.DeepCopyInto(out)
return out
}
// DeepCopyInto is an autogenerated deepcopy function, copying the receiver, writing into out. in must be non-nil.
-func (in *LogicalConfig) DeepCopyInto(out *LogicalConfig) {
+func (in *RestoreList) DeepCopyInto(out *RestoreList) {
*out = *in
- in.BackupToolRestoreCommand.DeepCopyInto(&out.BackupToolRestoreCommand)
+ out.TypeMeta = in.TypeMeta
+ in.ListMeta.DeepCopyInto(&out.ListMeta)
+ if in.Items != nil {
+ in, out := &in.Items, &out.Items
+ *out = make([]Restore, len(*in))
+ for i := range *in {
+ (*in)[i].DeepCopyInto(&(*out)[i])
+ }
+ }
}
-// DeepCopy is an autogenerated deepcopy function, copying the receiver, creating a new LogicalConfig.
-func (in *LogicalConfig) DeepCopy() *LogicalConfig {
+// DeepCopy is an autogenerated deepcopy function, copying the receiver, creating a new RestoreList.
+func (in *RestoreList) DeepCopy() *RestoreList {
if in == nil {
return nil
}
- out := new(LogicalConfig)
+ out := new(RestoreList)
in.DeepCopyInto(out)
return out
}
+// DeepCopyObject is an autogenerated deepcopy function, copying the receiver, creating a new runtime.Object.
+func (in *RestoreList) DeepCopyObject() runtime.Object {
+ if c := in.DeepCopy(); c != nil {
+ return c
+ }
+ return nil
+}
+
// DeepCopyInto is an autogenerated deepcopy function, copying the receiver, writing into out. in must be non-nil.
-func (in *ManifestsStatus) DeepCopyInto(out *ManifestsStatus) {
+func (in *RestoreSpec) DeepCopyInto(out *RestoreSpec) {
*out = *in
- if in.BackupLog != nil {
- in, out := &in.BackupLog, &out.BackupLog
- *out = new(BackupLogStatus)
+ out.Backup = in.Backup
+ if in.Resources != nil {
+ in, out := &in.Resources, &out.Resources
+ *out = new(RestoreKubeResources)
(*in).DeepCopyInto(*out)
}
- if in.Snapshot != nil {
- in, out := &in.Snapshot, &out.Snapshot
- *out = new(BackupSnapshotStatus)
- **out = **in
+ if in.PrepareDataConfig != nil {
+ in, out := &in.PrepareDataConfig, &out.PrepareDataConfig
+ *out = new(PrepareDataConfig)
+ (*in).DeepCopyInto(*out)
}
- if in.BackupTool != nil {
- in, out := &in.BackupTool, &out.BackupTool
- *out = new(BackupToolManifestsStatus)
- **out = **in
+ if in.ReadyConfig != nil {
+ in, out := &in.ReadyConfig, &out.ReadyConfig
+ *out = new(ReadyConfig)
+ (*in).DeepCopyInto(*out)
}
- if in.UserContext != nil {
- in, out := &in.UserContext, &out.UserContext
- *out = make(map[string]string, len(*in))
- for key, val := range *in {
- (*out)[key] = val
+ if in.Env != nil {
+ in, out := &in.Env, &out.Env
+ *out = make([]v1.EnvVar, len(*in))
+ for i := range *in {
+ (*in)[i].DeepCopyInto(&(*out)[i])
}
}
+ in.ContainerResources.DeepCopyInto(&out.ContainerResources)
}
-// DeepCopy is an autogenerated deepcopy function, copying the receiver, creating a new ManifestsStatus.
-func (in *ManifestsStatus) DeepCopy() *ManifestsStatus {
+// DeepCopy is an autogenerated deepcopy function, copying the receiver, creating a new RestoreSpec.
+func (in *RestoreSpec) DeepCopy() *RestoreSpec {
if in == nil {
return nil
}
- out := new(ManifestsStatus)
+ out := new(RestoreSpec)
in.DeepCopyInto(out)
return out
}
// DeepCopyInto is an autogenerated deepcopy function, copying the receiver, writing into out. in must be non-nil.
-func (in *PersistentVolumeClaim) DeepCopyInto(out *PersistentVolumeClaim) {
+func (in *RestoreStatus) DeepCopyInto(out *RestoreStatus) {
*out = *in
- if in.Name != nil {
- in, out := &in.Name, &out.Name
- *out = new(string)
- **out = **in
+ if in.StartTimestamp != nil {
+ in, out := &in.StartTimestamp, &out.StartTimestamp
+ *out = (*in).DeepCopy()
}
- if in.StorageClassName != nil {
- in, out := &in.StorageClassName, &out.StorageClassName
- *out = new(string)
- **out = **in
+ if in.CompletionTimestamp != nil {
+ in, out := &in.CompletionTimestamp, &out.CompletionTimestamp
+ *out = (*in).DeepCopy()
}
- out.InitCapacity = in.InitCapacity.DeepCopy()
- if in.PersistentVolumeConfigMap != nil {
- in, out := &in.PersistentVolumeConfigMap, &out.PersistentVolumeConfigMap
- *out = new(PersistentVolumeConfigMap)
+ if in.Duration != nil {
+ in, out := &in.Duration, &out.Duration
+ *out = new(metav1.Duration)
**out = **in
}
+ in.Actions.DeepCopyInto(&out.Actions)
+ if in.Conditions != nil {
+ in, out := &in.Conditions, &out.Conditions
+ *out = make([]metav1.Condition, len(*in))
+ for i := range *in {
+ (*in)[i].DeepCopyInto(&(*out)[i])
+ }
+ }
}
-// DeepCopy is an autogenerated deepcopy function, copying the receiver, creating a new PersistentVolumeClaim.
-func (in *PersistentVolumeClaim) DeepCopy() *PersistentVolumeClaim {
+// DeepCopy is an autogenerated deepcopy function, copying the receiver, creating a new RestoreStatus.
+func (in *RestoreStatus) DeepCopy() *RestoreStatus {
if in == nil {
return nil
}
- out := new(PersistentVolumeClaim)
+ out := new(RestoreStatus)
in.DeepCopyInto(out)
return out
}
// DeepCopyInto is an autogenerated deepcopy function, copying the receiver, writing into out. in must be non-nil.
-func (in *PersistentVolumeConfigMap) DeepCopyInto(out *PersistentVolumeConfigMap) {
+func (in *RestoreStatusAction) DeepCopyInto(out *RestoreStatusAction) {
*out = *in
+ in.StartTime.DeepCopyInto(&out.StartTime)
+ in.EndTime.DeepCopyInto(&out.EndTime)
}
-// DeepCopy is an autogenerated deepcopy function, copying the receiver, creating a new PersistentVolumeConfigMap.
-func (in *PersistentVolumeConfigMap) DeepCopy() *PersistentVolumeConfigMap {
+// DeepCopy is an autogenerated deepcopy function, copying the receiver, creating a new RestoreStatusAction.
+func (in *RestoreStatusAction) DeepCopy() *RestoreStatusAction {
if in == nil {
return nil
}
- out := new(PersistentVolumeConfigMap)
+ out := new(RestoreStatusAction)
in.DeepCopyInto(out)
return out
}
// DeepCopyInto is an autogenerated deepcopy function, copying the receiver, writing into out. in must be non-nil.
-func (in *PhysicalConfig) DeepCopyInto(out *PhysicalConfig) {
+func (in *RestoreStatusActions) DeepCopyInto(out *RestoreStatusActions) {
*out = *in
- in.BackupToolRestoreCommand.DeepCopyInto(&out.BackupToolRestoreCommand)
+ if in.PrepareData != nil {
+ in, out := &in.PrepareData, &out.PrepareData
+ *out = make([]RestoreStatusAction, len(*in))
+ for i := range *in {
+ (*in)[i].DeepCopyInto(&(*out)[i])
+ }
+ }
+ if in.PostReady != nil {
+ in, out := &in.PostReady, &out.PostReady
+ *out = make([]RestoreStatusAction, len(*in))
+ for i := range *in {
+ (*in)[i].DeepCopyInto(&(*out)[i])
+ }
+ }
}
-// DeepCopy is an autogenerated deepcopy function, copying the receiver, creating a new PhysicalConfig.
-func (in *PhysicalConfig) DeepCopy() *PhysicalConfig {
+// DeepCopy is an autogenerated deepcopy function, copying the receiver, creating a new RestoreStatusActions.
+func (in *RestoreStatusActions) DeepCopy() *RestoreStatusActions {
if in == nil {
return nil
}
- out := new(PhysicalConfig)
+ out := new(RestoreStatusActions)
in.DeepCopyInto(out)
return out
}
// DeepCopyInto is an autogenerated deepcopy function, copying the receiver, writing into out. in must be non-nil.
-func (in *RestoreJob) DeepCopyInto(out *RestoreJob) {
+func (in *RestoreVolumeClaim) DeepCopyInto(out *RestoreVolumeClaim) {
*out = *in
- out.TypeMeta = in.TypeMeta
in.ObjectMeta.DeepCopyInto(&out.ObjectMeta)
- in.Spec.DeepCopyInto(&out.Spec)
- in.Status.DeepCopyInto(&out.Status)
+ in.VolumeClaimSpec.DeepCopyInto(&out.VolumeClaimSpec)
+ out.VolumeConfig = in.VolumeConfig
}
-// DeepCopy is an autogenerated deepcopy function, copying the receiver, creating a new RestoreJob.
-func (in *RestoreJob) DeepCopy() *RestoreJob {
+// DeepCopy is an autogenerated deepcopy function, copying the receiver, creating a new RestoreVolumeClaim.
+func (in *RestoreVolumeClaim) DeepCopy() *RestoreVolumeClaim {
if in == nil {
return nil
}
- out := new(RestoreJob)
+ out := new(RestoreVolumeClaim)
in.DeepCopyInto(out)
return out
}
-// DeepCopyObject is an autogenerated deepcopy function, copying the receiver, creating a new runtime.Object.
-func (in *RestoreJob) DeepCopyObject() runtime.Object {
- if c := in.DeepCopy(); c != nil {
- return c
- }
- return nil
-}
-
// DeepCopyInto is an autogenerated deepcopy function, copying the receiver, writing into out. in must be non-nil.
-func (in *RestoreJobList) DeepCopyInto(out *RestoreJobList) {
+func (in *RestoreVolumeClaimsTemplate) DeepCopyInto(out *RestoreVolumeClaimsTemplate) {
*out = *in
- out.TypeMeta = in.TypeMeta
- in.ListMeta.DeepCopyInto(&out.ListMeta)
- if in.Items != nil {
- in, out := &in.Items, &out.Items
- *out = make([]RestoreJob, len(*in))
+ if in.Templates != nil {
+ in, out := &in.Templates, &out.Templates
+ *out = make([]RestoreVolumeClaim, len(*in))
for i := range *in {
(*in)[i].DeepCopyInto(&(*out)[i])
}
}
}
-// DeepCopy is an autogenerated deepcopy function, copying the receiver, creating a new RestoreJobList.
-func (in *RestoreJobList) DeepCopy() *RestoreJobList {
+// DeepCopy is an autogenerated deepcopy function, copying the receiver, creating a new RestoreVolumeClaimsTemplate.
+func (in *RestoreVolumeClaimsTemplate) DeepCopy() *RestoreVolumeClaimsTemplate {
if in == nil {
return nil
}
- out := new(RestoreJobList)
+ out := new(RestoreVolumeClaimsTemplate)
in.DeepCopyInto(out)
return out
}
-// DeepCopyObject is an autogenerated deepcopy function, copying the receiver, creating a new runtime.Object.
-func (in *RestoreJobList) DeepCopyObject() runtime.Object {
- if c := in.DeepCopy(); c != nil {
- return c
+// DeepCopyInto is an autogenerated deepcopy function, copying the receiver, writing into out. in must be non-nil.
+func (in *RuntimeSettings) DeepCopyInto(out *RuntimeSettings) {
+ *out = *in
+ in.Resources.DeepCopyInto(&out.Resources)
+}
+
+// DeepCopy is an autogenerated deepcopy function, copying the receiver, creating a new RuntimeSettings.
+func (in *RuntimeSettings) DeepCopy() *RuntimeSettings {
+ if in == nil {
+ return nil
}
- return nil
+ out := new(RuntimeSettings)
+ in.DeepCopyInto(out)
+ return out
}
// DeepCopyInto is an autogenerated deepcopy function, copying the receiver, writing into out. in must be non-nil.
-func (in *RestoreJobSpec) DeepCopyInto(out *RestoreJobSpec) {
+func (in *SchedulePolicy) DeepCopyInto(out *SchedulePolicy) {
*out = *in
- in.Target.DeepCopyInto(&out.Target)
- if in.TargetVolumes != nil {
- in, out := &in.TargetVolumes, &out.TargetVolumes
- *out = make([]corev1.Volume, len(*in))
- for i := range *in {
- (*in)[i].DeepCopyInto(&(*out)[i])
- }
- }
- if in.TargetVolumeMounts != nil {
- in, out := &in.TargetVolumeMounts, &out.TargetVolumeMounts
- *out = make([]corev1.VolumeMount, len(*in))
- for i := range *in {
- (*in)[i].DeepCopyInto(&(*out)[i])
- }
+ if in.Enabled != nil {
+ in, out := &in.Enabled, &out.Enabled
+ *out = new(bool)
+ **out = **in
}
}
-// DeepCopy is an autogenerated deepcopy function, copying the receiver, creating a new RestoreJobSpec.
-func (in *RestoreJobSpec) DeepCopy() *RestoreJobSpec {
+// DeepCopy is an autogenerated deepcopy function, copying the receiver, creating a new SchedulePolicy.
+func (in *SchedulePolicy) DeepCopy() *SchedulePolicy {
if in == nil {
return nil
}
- out := new(RestoreJobSpec)
+ out := new(SchedulePolicy)
in.DeepCopyInto(out)
return out
}
// DeepCopyInto is an autogenerated deepcopy function, copying the receiver, writing into out. in must be non-nil.
-func (in *RestoreJobStatus) DeepCopyInto(out *RestoreJobStatus) {
+func (in *ScheduleStatus) DeepCopyInto(out *ScheduleStatus) {
*out = *in
- if in.Expiration != nil {
- in, out := &in.Expiration, &out.Expiration
- *out = (*in).DeepCopy()
- }
- if in.StartTimestamp != nil {
- in, out := &in.StartTimestamp, &out.StartTimestamp
+ if in.LastScheduleTime != nil {
+ in, out := &in.LastScheduleTime, &out.LastScheduleTime
*out = (*in).DeepCopy()
}
- if in.CompletionTimestamp != nil {
- in, out := &in.CompletionTimestamp, &out.CompletionTimestamp
+ if in.LastSuccessfulTime != nil {
+ in, out := &in.LastSuccessfulTime, &out.LastSuccessfulTime
*out = (*in).DeepCopy()
}
}
-// DeepCopy is an autogenerated deepcopy function, copying the receiver, creating a new RestoreJobStatus.
-func (in *RestoreJobStatus) DeepCopy() *RestoreJobStatus {
+// DeepCopy is an autogenerated deepcopy function, copying the receiver, creating a new ScheduleStatus.
+func (in *ScheduleStatus) DeepCopy() *ScheduleStatus {
if in == nil {
return nil
}
- out := new(RestoreJobStatus)
+ out := new(ScheduleStatus)
in.DeepCopyInto(out)
return out
}
// DeepCopyInto is an autogenerated deepcopy function, copying the receiver, writing into out. in must be non-nil.
-func (in *RetentionSpec) DeepCopyInto(out *RetentionSpec) {
+func (in *SchedulingSpec) DeepCopyInto(out *SchedulingSpec) {
*out = *in
- if in.TTL != nil {
- in, out := &in.TTL, &out.TTL
- *out = new(string)
- **out = **in
+ if in.Tolerations != nil {
+ in, out := &in.Tolerations, &out.Tolerations
+ *out = make([]v1.Toleration, len(*in))
+ for i := range *in {
+ (*in)[i].DeepCopyInto(&(*out)[i])
+ }
+ }
+ if in.NodeSelector != nil {
+ in, out := &in.NodeSelector, &out.NodeSelector
+ *out = make(map[string]string, len(*in))
+ for key, val := range *in {
+ (*out)[key] = val
+ }
+ }
+ if in.Affinity != nil {
+ in, out := &in.Affinity, &out.Affinity
+ *out = new(v1.Affinity)
+ (*in).DeepCopyInto(*out)
+ }
+ if in.TopologySpreadConstraints != nil {
+ in, out := &in.TopologySpreadConstraints, &out.TopologySpreadConstraints
+ *out = make([]v1.TopologySpreadConstraint, len(*in))
+ for i := range *in {
+ (*in)[i].DeepCopyInto(&(*out)[i])
+ }
}
}
-// DeepCopy is an autogenerated deepcopy function, copying the receiver, creating a new RetentionSpec.
-func (in *RetentionSpec) DeepCopy() *RetentionSpec {
+// DeepCopy is an autogenerated deepcopy function, copying the receiver, creating a new SchedulingSpec.
+func (in *SchedulingSpec) DeepCopy() *SchedulingSpec {
if in == nil {
return nil
}
- out := new(RetentionSpec)
+ out := new(SchedulingSpec)
in.DeepCopyInto(out)
return out
}
// DeepCopyInto is an autogenerated deepcopy function, copying the receiver, writing into out. in must be non-nil.
-func (in *Schedule) DeepCopyInto(out *Schedule) {
+func (in *SyncProgress) DeepCopyInto(out *SyncProgress) {
*out = *in
- if in.StartingDeadlineMinutes != nil {
- in, out := &in.StartingDeadlineMinutes, &out.StartingDeadlineMinutes
- *out = new(int64)
- **out = **in
- }
- if in.Snapshot != nil {
- in, out := &in.Snapshot, &out.Snapshot
- *out = new(SchedulePolicy)
+ if in.Enabled != nil {
+ in, out := &in.Enabled, &out.Enabled
+ *out = new(bool)
**out = **in
}
- if in.Datafile != nil {
- in, out := &in.Datafile, &out.Datafile
- *out = new(SchedulePolicy)
- **out = **in
- }
- if in.Logfile != nil {
- in, out := &in.Logfile, &out.Logfile
- *out = new(SchedulePolicy)
+ if in.IntervalSeconds != nil {
+ in, out := &in.IntervalSeconds, &out.IntervalSeconds
+ *out = new(int32)
**out = **in
}
}
-// DeepCopy is an autogenerated deepcopy function, copying the receiver, creating a new Schedule.
-func (in *Schedule) DeepCopy() *Schedule {
+// DeepCopy is an autogenerated deepcopy function, copying the receiver, creating a new SyncProgress.
+func (in *SyncProgress) DeepCopy() *SyncProgress {
if in == nil {
return nil
}
- out := new(Schedule)
+ out := new(SyncProgress)
in.DeepCopyInto(out)
return out
}
// DeepCopyInto is an autogenerated deepcopy function, copying the receiver, writing into out. in must be non-nil.
-func (in *SchedulePolicy) DeepCopyInto(out *SchedulePolicy) {
+func (in *TargetVolumeInfo) DeepCopyInto(out *TargetVolumeInfo) {
*out = *in
+ if in.Volumes != nil {
+ in, out := &in.Volumes, &out.Volumes
+ *out = make([]string, len(*in))
+ copy(*out, *in)
+ }
+ if in.VolumeMounts != nil {
+ in, out := &in.VolumeMounts, &out.VolumeMounts
+ *out = make([]v1.VolumeMount, len(*in))
+ for i := range *in {
+ (*in)[i].DeepCopyInto(&(*out)[i])
+ }
+ }
}
-// DeepCopy is an autogenerated deepcopy function, copying the receiver, creating a new SchedulePolicy.
-func (in *SchedulePolicy) DeepCopy() *SchedulePolicy {
+// DeepCopy is an autogenerated deepcopy function, copying the receiver, creating a new TargetVolumeInfo.
+func (in *TargetVolumeInfo) DeepCopy() *TargetVolumeInfo {
if in == nil {
return nil
}
- out := new(SchedulePolicy)
+ out := new(TargetVolumeInfo)
in.DeepCopyInto(out)
return out
}
// DeepCopyInto is an autogenerated deepcopy function, copying the receiver, writing into out. in must be non-nil.
-func (in *SnapshotPolicy) DeepCopyInto(out *SnapshotPolicy) {
+func (in *VolumeConfig) DeepCopyInto(out *VolumeConfig) {
*out = *in
- in.BasePolicy.DeepCopyInto(&out.BasePolicy)
- if in.Hooks != nil {
- in, out := &in.Hooks, &out.Hooks
- *out = new(BackupPolicyHook)
- (*in).DeepCopyInto(*out)
- }
}
-// DeepCopy is an autogenerated deepcopy function, copying the receiver, creating a new SnapshotPolicy.
-func (in *SnapshotPolicy) DeepCopy() *SnapshotPolicy {
+// DeepCopy is an autogenerated deepcopy function, copying the receiver, creating a new VolumeConfig.
+func (in *VolumeConfig) DeepCopy() *VolumeConfig {
if in == nil {
return nil
}
- out := new(SnapshotPolicy)
+ out := new(VolumeConfig)
in.DeepCopyInto(out)
return out
}
// DeepCopyInto is an autogenerated deepcopy function, copying the receiver, writing into out. in must be non-nil.
-func (in *TargetCluster) DeepCopyInto(out *TargetCluster) {
+func (in *VolumeSnapshotStatus) DeepCopyInto(out *VolumeSnapshotStatus) {
*out = *in
- if in.LabelsSelector != nil {
- in, out := &in.LabelsSelector, &out.LabelsSelector
- *out = new(v1.LabelSelector)
- (*in).DeepCopyInto(*out)
- }
- if in.Secret != nil {
- in, out := &in.Secret, &out.Secret
- *out = new(BackupPolicySecret)
- **out = **in
- }
}
-// DeepCopy is an autogenerated deepcopy function, copying the receiver, creating a new TargetCluster.
-func (in *TargetCluster) DeepCopy() *TargetCluster {
+// DeepCopy is an autogenerated deepcopy function, copying the receiver, creating a new VolumeSnapshotStatus.
+func (in *VolumeSnapshotStatus) DeepCopy() *VolumeSnapshotStatus {
if in == nil {
return nil
}
- out := new(TargetCluster)
+ out := new(VolumeSnapshotStatus)
in.DeepCopyInto(out)
return out
}
diff --git a/apis/extensions/v1alpha1/addon_types.go b/apis/extensions/v1alpha1/addon_types.go
index eb1c18b5d2f..bbee2108b9a 100644
--- a/apis/extensions/v1alpha1/addon_types.go
+++ b/apis/extensions/v1alpha1/addon_types.go
@@ -92,9 +92,10 @@ type InstallableSpec struct {
}
type SelectorRequirement struct {
- // The selector key. Valid values are KubeVersion, KubeGitVersion.
+ // The selector key. Valid values are KubeVersion, KubeGitVersion and KubeProvider.
// "KubeVersion" the semver expression of Kubernetes versions, i.e., v1.24.
// "KubeGitVersion" may contain distro. info., i.e., v1.24.4+eks.
+ // "KubeProvider" the Kubernetes provider, i.e., aws,gcp,azure,huaweiCloud,tencentCloud etc.
// +kubebuilder:validation:Required
Key AddonSelectorKey `json:"key"`
@@ -494,6 +495,8 @@ func (r SelectorRequirement) MatchesFromConfig() bool {
l = ver.GitVersion
case KubeVersion:
l = fmt.Sprintf("%s.%s", ver.Major, ver.Minor)
+ case KubeProvider:
+ l = viper.GetString(constant.CfgKeyProvider)
}
return r.matchesLine(l)
}
diff --git a/apis/extensions/v1alpha1/type.go b/apis/extensions/v1alpha1/type.go
index a213763a559..2c039ac17ad 100644
--- a/apis/extensions/v1alpha1/type.go
+++ b/apis/extensions/v1alpha1/type.go
@@ -51,12 +51,13 @@ const (
// AddonSelectorKey are selector requirement key types.
// +enum
-// +kubebuilder:validation:Enum={KubeGitVersion,KubeVersion}
+// +kubebuilder:validation:Enum={KubeGitVersion,KubeVersion,KubeProvider}
type AddonSelectorKey string
const (
KubeGitVersion AddonSelectorKey = "KubeGitVersion"
KubeVersion AddonSelectorKey = "KubeVersion"
+ KubeProvider AddonSelectorKey = "KubeProvider"
)
const (
diff --git a/apis/storage/v1alpha1/storageprovider_types.go b/apis/storage/v1alpha1/storageprovider_types.go
index 1a5026fc3c6..daa85e6ed4a 100644
--- a/apis/storage/v1alpha1/storageprovider_types.go
+++ b/apis/storage/v1alpha1/storageprovider_types.go
@@ -37,11 +37,25 @@ type StorageProviderSpec struct {
// The template will be rendered with the following variables:
// - Parameters: a map of parameters defined in the ParametersSchema.
// - CSIDriverSecretRef: the reference of the secret created by the CSIDriverSecretTemplate.
- // +kubebuilder:validation:Required
+ // +optional
StorageClassTemplate string `json:"storageClassTemplate,omitempty"`
+ // A Go template for rendering a PersistentVolumeClaim.
+ // The template will be rendered with the following variables:
+ // - Parameters: a map of parameters defined in the ParametersSchema.
+ // - GeneratedStorageClassName: the name of the storage class generated with the StorageClassTemplate.
+ // +optional
+ PersistentVolumeClaimTemplate string `json:"persistentVolumeClaimTemplate,omitempty"`
+
+ // A Go template for rendering a config used by the datasafed command.
+ // The template will be rendered with the following variables:
+ // - Parameters: a map of parameters defined in the ParametersSchema.
+ // +optional
+ DatasafedConfigTemplate string `json:"datasafedConfigTemplate,omitempty"`
+
// The schema describes the parameters required by this StorageProvider,
// when rendering the templates.
+ // +optional
ParametersSchema *ParametersSchema `json:"parametersSchema,omitempty"`
}
@@ -52,6 +66,7 @@ type ParametersSchema struct {
// +kubebuilder:validation:Type=object
// +kubebuilder:pruning:PreserveUnknownFields
// +k8s:conversion-gen=false
+ // +optional
OpenAPIV3Schema *apiextensionsv1.JSONSchemaProps `json:"openAPIV3Schema,omitempty"`
// credentialFields are the fields used to generate the secret.
@@ -76,6 +91,9 @@ type StorageProviderStatus struct {
// +kubebuilder:object:root=true
// +kubebuilder:subresource:status
// +kubebuilder:resource:categories={kubeblocks},scope=Cluster
+// +kubebuilder:printcolumn:name="STATUS",type="string",JSONPath=".status.phase"
+// +kubebuilder:printcolumn:name="CSIDRIVER",type="string",JSONPath=".spec.csiDriverName"
+// +kubebuilder:printcolumn:name="AGE",type="date",JSONPath=".metadata.creationTimestamp"
// StorageProvider is the Schema for the storageproviders API
// StorageProvider describes how to provision PVCs for a specific storage system (e.g. S3, NFS, etc),
diff --git a/apis/workloads/v1alpha1/replicatedstatemachine_types.go b/apis/workloads/v1alpha1/replicatedstatemachine_types.go
index 013aed80aec..9f1fdd2146b 100644
--- a/apis/workloads/v1alpha1/replicatedstatemachine_types.go
+++ b/apis/workloads/v1alpha1/replicatedstatemachine_types.go
@@ -149,7 +149,7 @@ type ReplicatedStateMachineStatus struct {
// +kubebuilder:printcolumn:name="REPLICAS",type="string",JSONPath=".status.replicas",description="total replicas."
// +kubebuilder:printcolumn:name="AGE",type="date",JSONPath=".metadata.creationTimestamp"
-// ReplicatedStateMachine is the Schema for the replicatedstatemachines API
+// ReplicatedStateMachine is the Schema for the replicatedstatemachines API.
type ReplicatedStateMachine struct {
metav1.TypeMeta `json:",inline"`
metav1.ObjectMeta `json:"metadata,omitempty"`
@@ -210,6 +210,16 @@ const (
ParallelUpdateStrategy MemberUpdateStrategy = "Parallel"
)
+// RoleUpdateMechanism defines the way how pod role label being updated.
+// +enum
+type RoleUpdateMechanism string
+
+const (
+ ReadinessProbeEventUpdate RoleUpdateMechanism = "ReadinessProbeEventUpdate"
+ DirectAPIServerEventUpdate RoleUpdateMechanism = "DirectAPIServerEventUpdate"
+ NoneUpdate RoleUpdateMechanism = "None"
+)
+
// RoleProbe defines how to observe role
type RoleProbe struct {
// ProbeActions define Actions to be taken in serial.
@@ -255,6 +265,12 @@ type RoleProbe struct {
// +kubebuilder:validation:Minimum=1
// +optional
FailureThreshold int32 `json:"failureThreshold,omitempty"`
+
+ // RoleUpdateMechanism specifies the way how pod role label being updated.
+ // +kubebuilder:default=None
+ // +kubebuilder:validation:Enum={ReadinessProbeEventUpdate, DirectAPIServerEventUpdate, None}
+ // +optional
+ RoleUpdateMechanism RoleUpdateMechanism `json:"roleUpdateMechanism,omitempty"`
}
type Credential struct {
diff --git a/apis/workloads/v1alpha1/replicatedstatemachine_webhook.go b/apis/workloads/v1alpha1/replicatedstatemachine_webhook.go
index 9247c1d3c55..52565c5ac05 100644
--- a/apis/workloads/v1alpha1/replicatedstatemachine_webhook.go
+++ b/apis/workloads/v1alpha1/replicatedstatemachine_webhook.go
@@ -27,6 +27,7 @@ import (
ctrl "sigs.k8s.io/controller-runtime"
logf "sigs.k8s.io/controller-runtime/pkg/log"
"sigs.k8s.io/controller-runtime/pkg/webhook"
+ "sigs.k8s.io/controller-runtime/pkg/webhook/admission"
)
// log is for logging in this package.
@@ -57,24 +58,24 @@ func (r *ReplicatedStateMachine) Default() {
var _ webhook.Validator = &ReplicatedStateMachine{}
// ValidateCreate implements webhook.Validator so a webhook will be registered for the type
-func (r *ReplicatedStateMachine) ValidateCreate() error {
+func (r *ReplicatedStateMachine) ValidateCreate() (admission.Warnings, error) {
replicatedstatemachinelog.Info("validate create", "name", r.Name)
- return r.validate()
+ return nil, r.validate()
}
// ValidateUpdate implements webhook.Validator so a webhook will be registered for the type
-func (r *ReplicatedStateMachine) ValidateUpdate(old runtime.Object) error {
+func (r *ReplicatedStateMachine) ValidateUpdate(old runtime.Object) (admission.Warnings, error) {
replicatedstatemachinelog.Info("validate update", "name", r.Name)
- return r.validate()
+ return nil, r.validate()
}
// ValidateDelete implements webhook.Validator so a webhook will be registered for the type
-func (r *ReplicatedStateMachine) ValidateDelete() error {
+func (r *ReplicatedStateMachine) ValidateDelete() (admission.Warnings, error) {
replicatedstatemachinelog.Info("validate delete", "name", r.Name)
- return r.validate()
+ return nil, r.validate()
}
func (r *ReplicatedStateMachine) validate() error {
diff --git a/cmd/dataprotection/main.go b/cmd/dataprotection/main.go
index efdfb091f2d..67f7fffbca3 100644
--- a/cmd/dataprotection/main.go
+++ b/cmd/dataprotection/main.go
@@ -197,48 +197,49 @@ func main() {
os.Exit(1)
}
- if err = (&dpcontrollers.BackupToolReconciler{
+ if err = (&dpcontrollers.ActionSetReconciler{
Client: mgr.GetClient(),
Scheme: mgr.GetScheme(),
- Recorder: mgr.GetEventRecorderFor("backup-tool-controller"),
+ Recorder: mgr.GetEventRecorderFor("actionset-controller"),
}).SetupWithManager(mgr); err != nil {
- setupLog.Error(err, "unable to create controller", "controller", "BackupTool")
+ setupLog.Error(err, "unable to create controller", "controller", "ActionSet")
os.Exit(1)
}
- if err = (&dpcontrollers.BackupPolicyReconciler{
- Client: mgr.GetClient(),
- Scheme: mgr.GetScheme(),
- Recorder: mgr.GetEventRecorderFor("backup-policy-controller"),
+ if err = (&dpcontrollers.BackupReconciler{
+ Client: mgr.GetClient(),
+ Scheme: mgr.GetScheme(),
+ Recorder: mgr.GetEventRecorderFor("backup-controller"),
+ RestConfig: mgr.GetConfig(),
}).SetupWithManager(mgr); err != nil {
- setupLog.Error(err, "unable to create controller", "controller", "BackupPolicy")
+ setupLog.Error(err, "unable to create controller", "controller", "Backup")
os.Exit(1)
}
- if err = (&dpcontrollers.CronJobReconciler{
+ if err = (&dpcontrollers.RestoreReconciler{
Client: mgr.GetClient(),
Scheme: mgr.GetScheme(),
- Recorder: mgr.GetEventRecorderFor("cronjob-controller"),
+ Recorder: mgr.GetEventRecorderFor("restore-controller"),
}).SetupWithManager(mgr); err != nil {
- setupLog.Error(err, "unable to create controller", "controller", "CronJob")
+ setupLog.Error(err, "unable to create controller", "controller", "Restore")
os.Exit(1)
}
- if err = (&dpcontrollers.BackupReconciler{
+ if err = (&dpcontrollers.BackupPolicyReconciler{
Client: mgr.GetClient(),
Scheme: mgr.GetScheme(),
- Recorder: mgr.GetEventRecorderFor("backup-controller"),
+ Recorder: mgr.GetEventRecorderFor("backup-policy-controller"),
}).SetupWithManager(mgr); err != nil {
- setupLog.Error(err, "unable to create controller", "controller", "Backup")
+ setupLog.Error(err, "unable to create controller", "controller", "BackupPolicy")
os.Exit(1)
}
- if err = (&dpcontrollers.RestoreJobReconciler{
+ if err = (&dpcontrollers.BackupScheduleReconciler{
Client: mgr.GetClient(),
Scheme: mgr.GetScheme(),
- Recorder: mgr.GetEventRecorderFor("restore-job-controller"),
+ Recorder: mgr.GetEventRecorderFor("backup-schedule-controller"),
}).SetupWithManager(mgr); err != nil {
- setupLog.Error(err, "unable to create controller", "controller", "RestoreJob")
+ setupLog.Error(err, "unable to create controller", "controller", "BackupSchedule")
os.Exit(1)
}
diff --git a/cmd/manager/main.go b/cmd/manager/main.go
index 78697f0c221..76706f68643 100644
--- a/cmd/manager/main.go
+++ b/cmd/manager/main.go
@@ -46,7 +46,7 @@ import (
// +kubebuilder:scaffold:imports
appsv1alpha1 "github.com/apecloud/kubeblocks/apis/apps/v1alpha1"
- dataprotectionv1alpha1 "github.com/apecloud/kubeblocks/apis/dataprotection/v1alpha1"
+ dpv1alpha1 "github.com/apecloud/kubeblocks/apis/dataprotection/v1alpha1"
extensionsv1alpha1 "github.com/apecloud/kubeblocks/apis/extensions/v1alpha1"
storagev1alpha1 "github.com/apecloud/kubeblocks/apis/storage/v1alpha1"
workloadsv1alpha1 "github.com/apecloud/kubeblocks/apis/workloads/v1alpha1"
@@ -59,7 +59,6 @@ import (
"github.com/apecloud/kubeblocks/internal/constant"
intctrlutil "github.com/apecloud/kubeblocks/internal/controllerutil"
viper "github.com/apecloud/kubeblocks/internal/viperx"
- "github.com/apecloud/kubeblocks/internal/webhook"
)
// added lease.coordination.k8s.io for leader election
@@ -78,7 +77,7 @@ func init() {
utilruntime.Must(clientgoscheme.AddToScheme(scheme))
utilruntime.Must(appsv1alpha1.AddToScheme(scheme))
- utilruntime.Must(dataprotectionv1alpha1.AddToScheme(scheme))
+ utilruntime.Must(dpv1alpha1.AddToScheme(scheme))
utilruntime.Must(snapshotv1.AddToScheme(scheme))
utilruntime.Must(snapshotv1beta1.AddToScheme(scheme))
utilruntime.Must(extensionsv1alpha1.AddToScheme(scheme))
@@ -457,11 +456,6 @@ func main() {
setupLog.Error(err, "unable to create webhook", "webhook", "ServiceDescriptor")
os.Exit(1)
}
-
- if err = webhook.SetupWithManager(mgr); err != nil {
- setupLog.Error(err, "unable to setup webhook")
- os.Exit(1)
- }
}
if err := mgr.AddHealthzCheck("healthz", healthz.Ping); err != nil {
diff --git a/config/crd/bases/apps.kubeblocks.io_backuppolicytemplates.yaml b/config/crd/bases/apps.kubeblocks.io_backuppolicytemplates.yaml
index ba3e79f8e16..2073e6f7e8e 100644
--- a/config/crd/bases/apps.kubeblocks.io_backuppolicytemplates.yaml
+++ b/config/crd/bases/apps.kubeblocks.io_backuppolicytemplates.yaml
@@ -2,8 +2,7 @@ apiVersion: apiextensions.k8s.io/v1
kind: CustomResourceDefinition
metadata:
annotations:
- controller-gen.kubebuilder.io/version: v0.9.0
- creationTimestamp: null
+ controller-gen.kubebuilder.io/version: v0.12.1
labels:
app.kubernetes.io/name: kubeblocks
name: backuppolicytemplates.apps.kubeblocks.io
@@ -54,6 +53,268 @@ spec:
the specified componentDefinition.
items:
properties:
+ backupMethods:
+ description: backupMethods defines the backup methods.
+ items:
+ description: BackupMethod defines the backup method.
+ properties:
+ actionSetName:
+ description: actionSetName refers to the ActionSet object
+ that defines the backup actions. For volume snapshot
+ backup, the actionSet is not required, the controller
+ will use the CSI volume snapshotter to create the snapshot.
+ type: string
+ env:
+ description: env specifies the environment variables for
+ the backup workload.
+ items:
+ description: EnvVar represents an environment variable
+ present in a Container.
+ properties:
+ name:
+ description: Name of the environment variable. Must
+ be a C_IDENTIFIER.
+ type: string
+ value:
+ description: 'Variable references $(VAR_NAME) are
+ expanded using the previously defined environment
+ variables in the container and any service environment
+ variables. If a variable cannot be resolved, the
+ reference in the input string will be unchanged.
+ Double $$ are reduced to a single $, which allows
+ for escaping the $(VAR_NAME) syntax: i.e. "$$(VAR_NAME)"
+ will produce the string literal "$(VAR_NAME)".
+ Escaped references will never be expanded, regardless
+ of whether the variable exists or not. Defaults
+ to "".'
+ type: string
+ valueFrom:
+ description: Source for the environment variable's
+ value. Cannot be used if value is not empty.
+ properties:
+ configMapKeyRef:
+ description: Selects a key of a ConfigMap.
+ properties:
+ key:
+ description: The key to select.
+ type: string
+ name:
+ description: 'Name of the referent. More
+ info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names
+ TODO: Add other useful fields. apiVersion,
+ kind, uid?'
+ type: string
+ optional:
+ description: Specify whether the ConfigMap
+ or its key must be defined
+ type: boolean
+ required:
+ - key
+ type: object
+ x-kubernetes-map-type: atomic
+ fieldRef:
+ description: 'Selects a field of the pod: supports
+ metadata.name, metadata.namespace, `metadata.labels['''']`,
+ `metadata.annotations['''']`, spec.nodeName,
+ spec.serviceAccountName, status.hostIP, status.podIP,
+ status.podIPs.'
+ properties:
+ apiVersion:
+ description: Version of the schema the FieldPath
+ is written in terms of, defaults to "v1".
+ type: string
+ fieldPath:
+ description: Path of the field to select
+ in the specified API version.
+ type: string
+ required:
+ - fieldPath
+ type: object
+ x-kubernetes-map-type: atomic
+ resourceFieldRef:
+ description: 'Selects a resource of the container:
+ only resources limits and requests (limits.cpu,
+ limits.memory, limits.ephemeral-storage, requests.cpu,
+ requests.memory and requests.ephemeral-storage)
+ are currently supported.'
+ properties:
+ containerName:
+ description: 'Container name: required for
+ volumes, optional for env vars'
+ type: string
+ divisor:
+ anyOf:
+ - type: integer
+ - type: string
+ description: Specifies the output format
+ of the exposed resources, defaults to
+ "1"
+ pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$
+ x-kubernetes-int-or-string: true
+ resource:
+ description: 'Required: resource to select'
+ type: string
+ required:
+ - resource
+ type: object
+ x-kubernetes-map-type: atomic
+ secretKeyRef:
+ description: Selects a key of a secret in the
+ pod's namespace
+ properties:
+ key:
+ description: The key of the secret to select
+ from. Must be a valid secret key.
+ type: string
+ name:
+ description: 'Name of the referent. More
+ info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names
+ TODO: Add other useful fields. apiVersion,
+ kind, uid?'
+ type: string
+ optional:
+ description: Specify whether the Secret
+ or its key must be defined
+ type: boolean
+ required:
+ - key
+ type: object
+ x-kubernetes-map-type: atomic
+ type: object
+ required:
+ - name
+ type: object
+ type: array
+ name:
+ description: the name of backup method.
+ pattern: ^[a-z0-9]([a-z0-9\.\-]*[a-z0-9])?$
+ type: string
+ runtimeSettings:
+ description: runtimeSettings specifies runtime settings
+ for the backup workload container.
+ properties:
+ resources:
+ description: 'resources specifies the resource required
+ by container. More info: https://kubernetes.io/docs/concepts/configuration/manage-compute-resources-container/'
+ properties:
+ claims:
+ description: "Claims lists the names of resources,
+ defined in spec.resourceClaims, that are used
+ by this container. \n This is an alpha field
+ and requires enabling the DynamicResourceAllocation
+ feature gate. \n This field is immutable. It
+ can only be set for containers."
+ items:
+ description: ResourceClaim references one entry
+ in PodSpec.ResourceClaims.
+ properties:
+ name:
+ description: Name must match the name of
+ one entry in pod.spec.resourceClaims of
+ the Pod where this field is used. It makes
+ that resource available inside a container.
+ type: string
+ required:
+ - name
+ type: object
+ type: array
+ x-kubernetes-list-map-keys:
+ - name
+ x-kubernetes-list-type: map
+ limits:
+ additionalProperties:
+ anyOf:
+ - type: integer
+ - type: string
+ pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$
+ x-kubernetes-int-or-string: true
+ description: 'Limits describes the maximum amount
+ of compute resources allowed. More info: https://kubernetes.io/docs/concepts/configuration/manage-resources-containers/'
+ type: object
+ requests:
+ additionalProperties:
+ anyOf:
+ - type: integer
+ - type: string
+ pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$
+ x-kubernetes-int-or-string: true
+ description: 'Requests describes the minimum amount
+ of compute resources required. If Requests is
+ omitted for a container, it defaults to Limits
+ if that is explicitly specified, otherwise to
+ an implementation-defined value. Requests cannot
+ exceed Limits. More info: https://kubernetes.io/docs/concepts/configuration/manage-resources-containers/'
+ type: object
+ type: object
+ type: object
+ snapshotVolumes:
+ default: false
+ description: snapshotVolumes specifies whether to take
+ snapshots of persistent volumes. if true, the BackupScript
+ is not required, the controller will use the CSI volume
+ snapshotter to create the snapshot.
+ type: boolean
+ targetVolumes:
+ description: targetVolumes specifies which volumes from
+ the target should be mounted in the backup workload.
+ properties:
+ volumeMounts:
+ description: volumeMounts specifies the mount for
+ the volumes specified in `Volumes` section.
+ items:
+ description: VolumeMount describes a mounting of
+ a Volume within a container.
+ properties:
+ mountPath:
+ description: Path within the container at which
+ the volume should be mounted. Must not contain
+ ':'.
+ type: string
+ mountPropagation:
+ description: mountPropagation determines how
+ mounts are propagated from the host to container
+ and the other way around. When not set, MountPropagationNone
+ is used. This field is beta in 1.10.
+ type: string
+ name:
+ description: This must match the Name of a Volume.
+ type: string
+ readOnly:
+ description: Mounted read-only if true, read-write
+ otherwise (false or unspecified). Defaults
+ to false.
+ type: boolean
+ subPath:
+ description: Path within the volume from which
+ the container's volume should be mounted.
+ Defaults to "" (volume's root).
+ type: string
+ subPathExpr:
+ description: Expanded path within the volume
+ from which the container's volume should be
+ mounted. Behaves similarly to SubPath but
+ environment variable references $(VAR_NAME)
+ are expanded using the container's environment.
+ Defaults to "" (volume's root). SubPathExpr
+ and SubPath are mutually exclusive.
+ type: string
+ required:
+ - mountPath
+ - name
+ type: object
+ type: array
+ volumes:
+ description: Volumes indicates the list of volumes
+ of targeted application that should be mounted on
+ the backup job.
+ items:
+ type: string
+ type: array
+ type: object
+ required:
+ - name
+ type: object
+ type: array
componentDefRef:
description: componentDefRef references componentDef defined
in ClusterDefinition spec. Need to comply with IANA Service
@@ -61,378 +322,86 @@ spec:
maxLength: 22
pattern: ^[a-z]([a-z0-9\-]*[a-z0-9])?$
type: string
- datafile:
- description: the policy for datafile backup.
+ retentionPeriod:
+ default: 7d
+ description: "retentionPeriod determines a duration up to which
+ the backup should be kept. controller will remove all backups
+ that are older than the RetentionPeriod. For example, RetentionPeriod
+ of `30d` will keep only the backups of last 30 days. Sample
+ duration format: - years: \t2y - months: \t6mo - days: \t\t30d
+ - hours: \t12h - minutes: \t30m You can also combine the above
+ durations. For example: 30d12h30m"
+ type: string
+ schedules:
+ description: schedule policy for backup.
+ items:
+ properties:
+ backupMethod:
+ description: backupMethod specifies the backup method
+ name that is defined in backupPolicy.
+ type: string
+ cronExpression:
+ description: the cron expression for schedule, the timezone
+ is in UTC. see https://en.wikipedia.org/wiki/Cron.
+ type: string
+ enabled:
+ description: enabled specifies whether the backup schedule
+ is enabled or not.
+ type: boolean
+ required:
+ - backupMethod
+ - cronExpression
+ type: object
+ type: array
+ target:
+ description: target instance for backup.
properties:
- backupStatusUpdates:
- description: define how to update metadata for backup status.
- items:
- properties:
- containerName:
- description: which container name that kubectl can
- execute.
- type: string
- path:
- description: 'specify the json path of backup object
- for patch. example: manifests.backupLog -- means
- patch the backup json path of status.manifests.backupLog.'
- type: string
- script:
- description: the shell Script commands to collect
- backup status metadata. The script must exist in
- the container of ContainerName and the output format
- must be set to JSON. Note that outputting to stderr
- may cause the result format to not be in JSON.
- type: string
- updateStage:
- description: 'when to update the backup status, pre:
- before backup, post: after backup'
- enum:
- - pre
- - post
- type: string
- useTargetPodServiceAccount:
- description: useTargetPodServiceAccount defines whether
- this job requires the service account of the backup
- target pod. if true, will use the service account
- of the backup target pod. otherwise, will use the
- system service account.
- type: boolean
- required:
- - updateStage
- type: object
- type: array
- backupToolName:
- description: which backup tool to perform database backup,
- only support one tool.
- pattern: ^[a-z0-9]([a-z0-9\.\-]*[a-z0-9])?$
+ account:
+ description: refer to spec.componentDef.systemAccounts.accounts[*].name
+ in ClusterDefinition. the secret created by this account
+ will be used to connect the database. if not set, the
+ secret created by spec.ConnectionCredential of the ClusterDefinition
+ will be used. it will be transformed to a secret for BackupPolicy's
+ target secret.
type: string
- backupsHistoryLimit:
- default: 7
- description: the number of automatic backups to retain.
- Value must be non-negative integer. 0 means NO limit on
- the number of backups.
- format: int32
- type: integer
- onFailAttempted:
- description: count of backup stop retries on fail.
- format: int32
- type: integer
- target:
- description: target instance for backup.
+ connectionCredentialKey:
+ description: connectionCredentialKey defines connection
+ credential key in secret which created by spec.ConnectionCredential
+ of the ClusterDefinition. it will be ignored when "account"
+ is set.
properties:
- account:
- description: refer to spec.componentDef.systemAccounts.accounts[*].name
- in ClusterDefinition. the secret created by this account
- will be used to connect the database. if not set,
- the secret created by spec.ConnectionCredential of
- the ClusterDefinition will be used. it will be transformed
- to a secret for BackupPolicy's target secret.
+ hostKey:
+ description: hostKey specifies the map key of the host
+ in the connection credential secret.
type: string
- connectionCredentialKey:
- description: connectionCredentialKey defines connection
- credential key in secret which created by spec.ConnectionCredential
- of the ClusterDefinition. it will be ignored when
- "account" is set.
- properties:
- passwordKey:
- description: the key of password in the ConnectionCredential
- secret. if not set, the default key is "password".
- type: string
- usernameKey:
- description: the key of username in the ConnectionCredential
- secret. if not set, the default key is "username".
- type: string
- type: object
- role:
- description: 'select instance of corresponding role
- for backup, role are: - the name of Leader/Follower/Leaner
- for Consensus component. - primary or secondary for
- Replication component. finally, invalid role of the
- component will be ignored. such as if workload type
- is Replication and component''s replicas is 1, the
- secondary role is invalid. and it also will be ignored
- when component is Stateful/Stateless. the role will
- be transformed to a role LabelSelector for BackupPolicy''s
- target attribute.'
+ passwordKey:
+ description: the key of password in the ConnectionCredential
+ secret. if not set, the default key is "password".
type: string
- type: object
- type: object
- logfile:
- description: the policy for logfile backup.
- properties:
- backupStatusUpdates:
- description: define how to update metadata for backup status.
- items:
- properties:
- containerName:
- description: which container name that kubectl can
- execute.
- type: string
- path:
- description: 'specify the json path of backup object
- for patch. example: manifests.backupLog -- means
- patch the backup json path of status.manifests.backupLog.'
- type: string
- script:
- description: the shell Script commands to collect
- backup status metadata. The script must exist in
- the container of ContainerName and the output format
- must be set to JSON. Note that outputting to stderr
- may cause the result format to not be in JSON.
- type: string
- updateStage:
- description: 'when to update the backup status, pre:
- before backup, post: after backup'
- enum:
- - pre
- - post
- type: string
- useTargetPodServiceAccount:
- description: useTargetPodServiceAccount defines whether
- this job requires the service account of the backup
- target pod. if true, will use the service account
- of the backup target pod. otherwise, will use the
- system service account.
- type: boolean
- required:
- - updateStage
- type: object
- type: array
- backupToolName:
- description: which backup tool to perform database backup,
- only support one tool.
- pattern: ^[a-z0-9]([a-z0-9\.\-]*[a-z0-9])?$
- type: string
- backupsHistoryLimit:
- default: 7
- description: the number of automatic backups to retain.
- Value must be non-negative integer. 0 means NO limit on
- the number of backups.
- format: int32
- type: integer
- onFailAttempted:
- description: count of backup stop retries on fail.
- format: int32
- type: integer
- target:
- description: target instance for backup.
- properties:
- account:
- description: refer to spec.componentDef.systemAccounts.accounts[*].name
- in ClusterDefinition. the secret created by this account
- will be used to connect the database. if not set,
- the secret created by spec.ConnectionCredential of
- the ClusterDefinition will be used. it will be transformed
- to a secret for BackupPolicy's target secret.
+ portKey:
+ default: port
+ description: portKey specifies the map key of the port
+ in the connection credential secret.
type: string
- connectionCredentialKey:
- description: connectionCredentialKey defines connection
- credential key in secret which created by spec.ConnectionCredential
- of the ClusterDefinition. it will be ignored when
- "account" is set.
- properties:
- passwordKey:
- description: the key of password in the ConnectionCredential
- secret. if not set, the default key is "password".
- type: string
- usernameKey:
- description: the key of username in the ConnectionCredential
- secret. if not set, the default key is "username".
- type: string
- type: object
- role:
- description: 'select instance of corresponding role
- for backup, role are: - the name of Leader/Follower/Leaner
- for Consensus component. - primary or secondary for
- Replication component. finally, invalid role of the
- component will be ignored. such as if workload type
- is Replication and component''s replicas is 1, the
- secondary role is invalid. and it also will be ignored
- when component is Stateful/Stateless. the role will
- be transformed to a role LabelSelector for BackupPolicy''s
- target attribute.'
+ usernameKey:
+ description: the key of username in the ConnectionCredential
+ secret. if not set, the default key is "username".
type: string
type: object
- type: object
- retention:
- description: retention describe how long the Backup should be
- retained. if not set, will be retained forever.
- properties:
- ttl:
- description: ttl is a time string ending with the 'd'|'D'|'h'|'H'
- character to describe how long the Backup should be retained.
- if not set, will be retained forever.
- pattern: ^\d+[d|D|h|H]$
+ role:
+ description: 'select instance of corresponding role for
+ backup, role are: - the name of Leader/Follower/Leaner
+ for Consensus component. - primary or secondary for Replication
+ component. finally, invalid role of the component will
+ be ignored. such as if workload type is Replication and
+ component''s replicas is 1, the secondary role is invalid.
+ and it also will be ignored when component is Stateful/Stateless.
+ the role will be transformed to a role LabelSelector for
+ BackupPolicy''s target attribute.'
type: string
type: object
- schedule:
- description: schedule policy for backup.
- properties:
- datafile:
- description: schedule policy for datafile backup.
- properties:
- cronExpression:
- description: the cron expression for schedule, the timezone
- is in UTC. see https://en.wikipedia.org/wiki/Cron.
- type: string
- enable:
- description: enable or disable the schedule.
- type: boolean
- required:
- - cronExpression
- - enable
- type: object
- logfile:
- description: schedule policy for logfile backup.
- properties:
- cronExpression:
- description: the cron expression for schedule, the timezone
- is in UTC. see https://en.wikipedia.org/wiki/Cron.
- type: string
- enable:
- description: enable or disable the schedule.
- type: boolean
- required:
- - cronExpression
- - enable
- type: object
- snapshot:
- description: schedule policy for snapshot backup.
- properties:
- cronExpression:
- description: the cron expression for schedule, the timezone
- is in UTC. see https://en.wikipedia.org/wiki/Cron.
- type: string
- enable:
- description: enable or disable the schedule.
- type: boolean
- required:
- - cronExpression
- - enable
- type: object
- startingDeadlineMinutes:
- description: startingDeadlineMinutes defines the deadline
- in minutes for starting the backup job if it misses scheduled
- time for any reason.
- format: int64
- maximum: 1440
- minimum: 0
- type: integer
- type: object
- snapshot:
- description: the policy for snapshot backup.
- properties:
- backupStatusUpdates:
- description: define how to update metadata for backup status.
- items:
- properties:
- containerName:
- description: which container name that kubectl can
- execute.
- type: string
- path:
- description: 'specify the json path of backup object
- for patch. example: manifests.backupLog -- means
- patch the backup json path of status.manifests.backupLog.'
- type: string
- script:
- description: the shell Script commands to collect
- backup status metadata. The script must exist in
- the container of ContainerName and the output format
- must be set to JSON. Note that outputting to stderr
- may cause the result format to not be in JSON.
- type: string
- updateStage:
- description: 'when to update the backup status, pre:
- before backup, post: after backup'
- enum:
- - pre
- - post
- type: string
- useTargetPodServiceAccount:
- description: useTargetPodServiceAccount defines whether
- this job requires the service account of the backup
- target pod. if true, will use the service account
- of the backup target pod. otherwise, will use the
- system service account.
- type: boolean
- required:
- - updateStage
- type: object
- type: array
- backupsHistoryLimit:
- default: 7
- description: the number of automatic backups to retain.
- Value must be non-negative integer. 0 means NO limit on
- the number of backups.
- format: int32
- type: integer
- hooks:
- description: execute hook commands for backup.
- properties:
- containerName:
- description: which container can exec command
- type: string
- image:
- description: exec command with image
- type: string
- postCommands:
- description: post backup to perform commands
- items:
- type: string
- type: array
- preCommands:
- description: pre backup to perform commands
- items:
- type: string
- type: array
- type: object
- onFailAttempted:
- description: count of backup stop retries on fail.
- format: int32
- type: integer
- target:
- description: target instance for backup.
- properties:
- account:
- description: refer to spec.componentDef.systemAccounts.accounts[*].name
- in ClusterDefinition. the secret created by this account
- will be used to connect the database. if not set,
- the secret created by spec.ConnectionCredential of
- the ClusterDefinition will be used. it will be transformed
- to a secret for BackupPolicy's target secret.
- type: string
- connectionCredentialKey:
- description: connectionCredentialKey defines connection
- credential key in secret which created by spec.ConnectionCredential
- of the ClusterDefinition. it will be ignored when
- "account" is set.
- properties:
- passwordKey:
- description: the key of password in the ConnectionCredential
- secret. if not set, the default key is "password".
- type: string
- usernameKey:
- description: the key of username in the ConnectionCredential
- secret. if not set, the default key is "username".
- type: string
- type: object
- role:
- description: 'select instance of corresponding role
- for backup, role are: - the name of Leader/Follower/Leaner
- for Consensus component. - primary or secondary for
- Replication component. finally, invalid role of the
- component will be ignored. such as if workload type
- is Replication and component''s replicas is 1, the
- secondary role is invalid. and it also will be ignored
- when component is Stateful/Stateless. the role will
- be transformed to a role LabelSelector for BackupPolicy''s
- target attribute.'
- type: string
- type: object
- type: object
required:
+ - backupMethods
- componentDefRef
type: object
minItems: 1
diff --git a/config/crd/bases/apps.kubeblocks.io_clusterdefinitions.yaml b/config/crd/bases/apps.kubeblocks.io_clusterdefinitions.yaml
index a86c3b22a9e..538dc69bc71 100644
--- a/config/crd/bases/apps.kubeblocks.io_clusterdefinitions.yaml
+++ b/config/crd/bases/apps.kubeblocks.io_clusterdefinitions.yaml
@@ -2,8 +2,7 @@ apiVersion: apiextensions.k8s.io/v1
kind: CustomResourceDefinition
metadata:
annotations:
- controller-gen.kubebuilder.io/version: v0.9.0
- creationTimestamp: null
+ controller-gen.kubebuilder.io/version: v0.12.1
labels:
app.kubernetes.io/name: kubeblocks
name: clusterdefinitions.apps.kubeblocks.io
@@ -700,6 +699,7 @@ spec:
type: object
type: array
type: object
+ x-kubernetes-map-type: atomic
weight:
description: Weight associated with matching
the corresponding nodeSelectorTerm, in the
@@ -810,10 +810,12 @@ spec:
type: object
type: array
type: object
+ x-kubernetes-map-type: atomic
type: array
required:
- nodeSelectorTerms
type: object
+ x-kubernetes-map-type: atomic
type: object
podAffinity:
description: Describes pod affinity scheduling rules
@@ -900,6 +902,7 @@ spec:
ANDed.
type: object
type: object
+ x-kubernetes-map-type: atomic
namespaceSelector:
description: A label query over the set
of namespaces that the term applies
@@ -963,6 +966,7 @@ spec:
ANDed.
type: object
type: object
+ x-kubernetes-map-type: atomic
namespaces:
description: namespaces specifies a static
list of namespace names that the term
@@ -1073,6 +1077,7 @@ spec:
only "value". The requirements are ANDed.
type: object
type: object
+ x-kubernetes-map-type: atomic
namespaceSelector:
description: A label query over the set of
namespaces that the term applies to. The
@@ -1131,6 +1136,7 @@ spec:
only "value". The requirements are ANDed.
type: object
type: object
+ x-kubernetes-map-type: atomic
namespaces:
description: namespaces specifies a static
list of namespace names that the term applies
@@ -1243,6 +1249,7 @@ spec:
ANDed.
type: object
type: object
+ x-kubernetes-map-type: atomic
namespaceSelector:
description: A label query over the set
of namespaces that the term applies
@@ -1306,6 +1313,7 @@ spec:
ANDed.
type: object
type: object
+ x-kubernetes-map-type: atomic
namespaces:
description: namespaces specifies a static
list of namespace names that the term
@@ -1417,6 +1425,7 @@ spec:
only "value". The requirements are ANDed.
type: object
type: object
+ x-kubernetes-map-type: atomic
namespaceSelector:
description: A label query over the set of
namespaces that the term applies to. The
@@ -1475,6 +1484,7 @@ spec:
only "value". The requirements are ANDed.
type: object
type: object
+ x-kubernetes-map-type: atomic
namespaces:
description: namespaces specifies a static
list of namespace names that the term applies
@@ -1594,6 +1604,7 @@ spec:
required:
- key
type: object
+ x-kubernetes-map-type: atomic
fieldRef:
description: 'Selects a field of the pod:
supports metadata.name, metadata.namespace,
@@ -1613,6 +1624,7 @@ spec:
required:
- fieldPath
type: object
+ x-kubernetes-map-type: atomic
resourceFieldRef:
description: 'Selects a resource of the
container: only resources limits and requests
@@ -1640,6 +1652,7 @@ spec:
required:
- resource
type: object
+ x-kubernetes-map-type: atomic
secretKeyRef:
description: Selects a key of a secret in
the pod's namespace
@@ -1662,6 +1675,7 @@ spec:
required:
- key
type: object
+ x-kubernetes-map-type: atomic
type: object
required:
- name
@@ -1694,6 +1708,7 @@ spec:
must be defined
type: boolean
type: object
+ x-kubernetes-map-type: atomic
prefix:
description: An optional identifier to prepend
to each key in the ConfigMap. Must be a C_IDENTIFIER.
@@ -1712,6 +1727,7 @@ spec:
must be defined
type: boolean
type: object
+ x-kubernetes-map-type: atomic
type: object
type: array
image:
@@ -1776,7 +1792,11 @@ spec:
custom header to be used in HTTP probes
properties:
name:
- description: The header field name
+ description: The header field name.
+ This will be canonicalized upon
+ output, so case-variant names
+ will be understood as the same
+ header.
type: string
value:
description: The header field value
@@ -1884,7 +1904,11 @@ spec:
custom header to be used in HTTP probes
properties:
name:
- description: The header field name
+ description: The header field name.
+ This will be canonicalized upon
+ output, so case-variant names
+ will be understood as the same
+ header.
type: string
value:
description: The header field value
@@ -1971,8 +1995,7 @@ spec:
type: integer
grpc:
description: GRPC specifies an action involving
- a GRPC port. This is a beta field and requires
- enabling GRPCContainerProbe feature gate.
+ a GRPC port.
properties:
port:
description: Port number of the gRPC service.
@@ -2006,7 +2029,10 @@ spec:
header to be used in HTTP probes
properties:
name:
- description: The header field name
+ description: The header field name.
+ This will be canonicalized upon output,
+ so case-variant names will be understood
+ as the same header.
type: string
value:
description: The header field value
@@ -2188,8 +2214,7 @@ spec:
type: integer
grpc:
description: GRPC specifies an action involving
- a GRPC port. This is a beta field and requires
- enabling GRPCContainerProbe feature gate.
+ a GRPC port.
properties:
port:
description: Port number of the gRPC service.
@@ -2223,7 +2248,10 @@ spec:
header to be used in HTTP probes
properties:
name:
- description: The header field name
+ description: The header field name.
+ This will be canonicalized upon output,
+ so case-variant names will be understood
+ as the same header.
type: string
value:
description: The header field value
@@ -2318,6 +2346,28 @@ spec:
format: int32
type: integer
type: object
+ resizePolicy:
+ description: Resources resize policy for the container.
+ items:
+ description: ContainerResizePolicy represents resource
+ resize policy for the container.
+ properties:
+ resourceName:
+ description: 'Name of the resource to which
+ this resource resize policy applies. Supported
+ values: cpu, memory.'
+ type: string
+ restartPolicy:
+ description: Restart policy to apply when specified
+ resource is resized. If not specified, it
+ defaults to NotRequired.
+ type: string
+ required:
+ - resourceName
+ - restartPolicy
+ type: object
+ type: array
+ x-kubernetes-list-type: atomic
resources:
description: 'Compute Resources required by this container.
Cannot be updated. More info: https://kubernetes.io/docs/concepts/configuration/manage-resources-containers/'
@@ -2367,10 +2417,33 @@ spec:
of compute resources required. If Requests is
omitted for a container, it defaults to Limits
if that is explicitly specified, otherwise to
- an implementation-defined value. More info:
- https://kubernetes.io/docs/concepts/configuration/manage-resources-containers/'
+ an implementation-defined value. Requests cannot
+ exceed Limits. More info: https://kubernetes.io/docs/concepts/configuration/manage-resources-containers/'
type: object
type: object
+ restartPolicy:
+ description: 'RestartPolicy defines the restart behavior
+ of individual containers in a pod. This field may
+ only be set for init containers, and the only allowed
+ value is "Always". For non-init containers or when
+ this field is not specified, the restart behavior
+ is defined by the Pod''s restart policy and the
+ container type. Setting the RestartPolicy as "Always"
+ for the init container will have the following effect:
+ this init container will be continually restarted
+ on exit until all regular containers have terminated.
+ Once all regular containers have completed, all
+ init containers with restartPolicy "Always" will
+ be shut down. This lifecycle differs from normal
+ init containers and is often referred to as a "sidecar"
+ container. Although this init container still starts
+ in the init container sequence, it does not wait
+ for the container to complete before proceeding
+ to the next init container. Instead, the next init
+ container starts immediately after this init container
+ is started, or after any startupProbe has successfully
+ completed.'
+ type: string
securityContext:
description: 'SecurityContext defines the security
options the container should be run with. If set,
@@ -2506,8 +2579,9 @@ spec:
be used. The profile must be preconfigured
on the node to work. Must be a descending
path, relative to the kubelet's configured
- seccomp profile location. Must only be set
- if type is "Localhost".
+ seccomp profile location. Must be set if
+ type is "Localhost". Must NOT be set for
+ any other type.
type: string
type:
description: "type indicates which kind of
@@ -2544,17 +2618,12 @@ spec:
hostProcess:
description: HostProcess determines if a container
should be run as a 'Host Process' container.
- This field is alpha-level and will only
- be honored by components that enable the
- WindowsHostProcessContainers feature flag.
- Setting this field without the feature flag
- will result in errors when validating the
- Pod. All of a Pod's containers must have
- the same effective HostProcess value (it
- is not allowed to have a mix of HostProcess
- containers and non-HostProcess containers). In
- addition, if HostProcess is true then HostNetwork
- must also be set to true.
+ All of a Pod's containers must have the
+ same effective HostProcess value (it is
+ not allowed to have a mix of HostProcess
+ containers and non-HostProcess containers).
+ In addition, if HostProcess is true then
+ HostNetwork must also be set to true.
type: boolean
runAsUserName:
description: The UserName in Windows to run
@@ -2605,8 +2674,7 @@ spec:
type: integer
grpc:
description: GRPC specifies an action involving
- a GRPC port. This is a beta field and requires
- enabling GRPCContainerProbe feature gate.
+ a GRPC port.
properties:
port:
description: Port number of the gRPC service.
@@ -2640,7 +2708,10 @@ spec:
header to be used in HTTP probes
properties:
name:
- description: The header field name
+ description: The header field name.
+ This will be canonicalized upon output,
+ so case-variant names will be understood
+ as the same header.
type: string
value:
description: The header field value
@@ -3009,6 +3080,7 @@ spec:
required:
- key
type: object
+ x-kubernetes-map-type: atomic
fieldRef:
description: 'Selects a field of the pod:
supports metadata.name, metadata.namespace,
@@ -3028,6 +3100,7 @@ spec:
required:
- fieldPath
type: object
+ x-kubernetes-map-type: atomic
resourceFieldRef:
description: 'Selects a resource of the
container: only resources limits and requests
@@ -3055,6 +3128,7 @@ spec:
required:
- resource
type: object
+ x-kubernetes-map-type: atomic
secretKeyRef:
description: Selects a key of a secret in
the pod's namespace
@@ -3077,6 +3151,7 @@ spec:
required:
- key
type: object
+ x-kubernetes-map-type: atomic
type: object
required:
- name
@@ -3109,6 +3184,7 @@ spec:
must be defined
type: boolean
type: object
+ x-kubernetes-map-type: atomic
prefix:
description: An optional identifier to prepend
to each key in the ConfigMap. Must be a C_IDENTIFIER.
@@ -3127,6 +3203,7 @@ spec:
must be defined
type: boolean
type: object
+ x-kubernetes-map-type: atomic
type: object
type: array
image:
@@ -3187,7 +3264,11 @@ spec:
custom header to be used in HTTP probes
properties:
name:
- description: The header field name
+ description: The header field name.
+ This will be canonicalized upon
+ output, so case-variant names
+ will be understood as the same
+ header.
type: string
value:
description: The header field value
@@ -3295,7 +3376,11 @@ spec:
custom header to be used in HTTP probes
properties:
name:
- description: The header field name
+ description: The header field name.
+ This will be canonicalized upon
+ output, so case-variant names
+ will be understood as the same
+ header.
type: string
value:
description: The header field value
@@ -3381,8 +3466,7 @@ spec:
type: integer
grpc:
description: GRPC specifies an action involving
- a GRPC port. This is a beta field and requires
- enabling GRPCContainerProbe feature gate.
+ a GRPC port.
properties:
port:
description: Port number of the gRPC service.
@@ -3416,7 +3500,10 @@ spec:
header to be used in HTTP probes
properties:
name:
- description: The header field name
+ description: The header field name.
+ This will be canonicalized upon output,
+ so case-variant names will be understood
+ as the same header.
type: string
value:
description: The header field value
@@ -3589,8 +3676,7 @@ spec:
type: integer
grpc:
description: GRPC specifies an action involving
- a GRPC port. This is a beta field and requires
- enabling GRPCContainerProbe feature gate.
+ a GRPC port.
properties:
port:
description: Port number of the gRPC service.
@@ -3624,7 +3710,10 @@ spec:
header to be used in HTTP probes
properties:
name:
- description: The header field name
+ description: The header field name.
+ This will be canonicalized upon output,
+ so case-variant names will be understood
+ as the same header.
type: string
value:
description: The header field value
@@ -3719,6 +3808,28 @@ spec:
format: int32
type: integer
type: object
+ resizePolicy:
+ description: Resources resize policy for the container.
+ items:
+ description: ContainerResizePolicy represents resource
+ resize policy for the container.
+ properties:
+ resourceName:
+ description: 'Name of the resource to which
+ this resource resize policy applies. Supported
+ values: cpu, memory.'
+ type: string
+ restartPolicy:
+ description: Restart policy to apply when specified
+ resource is resized. If not specified, it
+ defaults to NotRequired.
+ type: string
+ required:
+ - resourceName
+ - restartPolicy
+ type: object
+ type: array
+ x-kubernetes-list-type: atomic
resources:
description: Resources are not allowed for ephemeral
containers. Ephemeral containers use spare resources
@@ -3769,10 +3880,16 @@ spec:
of compute resources required. If Requests is
omitted for a container, it defaults to Limits
if that is explicitly specified, otherwise to
- an implementation-defined value. More info:
- https://kubernetes.io/docs/concepts/configuration/manage-resources-containers/'
+ an implementation-defined value. Requests cannot
+ exceed Limits. More info: https://kubernetes.io/docs/concepts/configuration/manage-resources-containers/'
type: object
type: object
+ restartPolicy:
+ description: Restart policy for the container to manage
+ the restart behavior of each container within a
+ pod. This may only be set for init containers. You
+ cannot set this field on ephemeral containers.
+ type: string
securityContext:
description: 'Optional: SecurityContext defines the
security options the ephemeral container should
@@ -3908,8 +4025,9 @@ spec:
be used. The profile must be preconfigured
on the node to work. Must be a descending
path, relative to the kubelet's configured
- seccomp profile location. Must only be set
- if type is "Localhost".
+ seccomp profile location. Must be set if
+ type is "Localhost". Must NOT be set for
+ any other type.
type: string
type:
description: "type indicates which kind of
@@ -3946,17 +4064,12 @@ spec:
hostProcess:
description: HostProcess determines if a container
should be run as a 'Host Process' container.
- This field is alpha-level and will only
- be honored by components that enable the
- WindowsHostProcessContainers feature flag.
- Setting this field without the feature flag
- will result in errors when validating the
- Pod. All of a Pod's containers must have
- the same effective HostProcess value (it
- is not allowed to have a mix of HostProcess
- containers and non-HostProcess containers). In
- addition, if HostProcess is true then HostNetwork
- must also be set to true.
+ All of a Pod's containers must have the
+ same effective HostProcess value (it is
+ not allowed to have a mix of HostProcess
+ containers and non-HostProcess containers).
+ In addition, if HostProcess is true then
+ HostNetwork must also be set to true.
type: boolean
runAsUserName:
description: The UserName in Windows to run
@@ -3999,8 +4112,7 @@ spec:
type: integer
grpc:
description: GRPC specifies an action involving
- a GRPC port. This is a beta field and requires
- enabling GRPCContainerProbe feature gate.
+ a GRPC port.
properties:
port:
description: Port number of the gRPC service.
@@ -4034,7 +4146,10 @@ spec:
header to be used in HTTP probes
properties:
name:
- description: The header field name
+ description: The header field name.
+ This will be canonicalized upon output,
+ so case-variant names will be understood
+ as the same header.
type: string
value:
description: The header field value
@@ -4331,6 +4446,7 @@ spec:
uid?'
type: string
type: object
+ x-kubernetes-map-type: atomic
type: array
initContainers:
description: 'List of initialization containers belonging
@@ -4430,6 +4546,7 @@ spec:
required:
- key
type: object
+ x-kubernetes-map-type: atomic
fieldRef:
description: 'Selects a field of the pod:
supports metadata.name, metadata.namespace,
@@ -4449,6 +4566,7 @@ spec:
required:
- fieldPath
type: object
+ x-kubernetes-map-type: atomic
resourceFieldRef:
description: 'Selects a resource of the
container: only resources limits and requests
@@ -4476,6 +4594,7 @@ spec:
required:
- resource
type: object
+ x-kubernetes-map-type: atomic
secretKeyRef:
description: Selects a key of a secret in
the pod's namespace
@@ -4498,6 +4617,7 @@ spec:
required:
- key
type: object
+ x-kubernetes-map-type: atomic
type: object
required:
- name
@@ -4530,6 +4650,7 @@ spec:
must be defined
type: boolean
type: object
+ x-kubernetes-map-type: atomic
prefix:
description: An optional identifier to prepend
to each key in the ConfigMap. Must be a C_IDENTIFIER.
@@ -4548,6 +4669,7 @@ spec:
must be defined
type: boolean
type: object
+ x-kubernetes-map-type: atomic
type: object
type: array
image:
@@ -4612,7 +4734,11 @@ spec:
custom header to be used in HTTP probes
properties:
name:
- description: The header field name
+ description: The header field name.
+ This will be canonicalized upon
+ output, so case-variant names
+ will be understood as the same
+ header.
type: string
value:
description: The header field value
@@ -4720,7 +4846,11 @@ spec:
custom header to be used in HTTP probes
properties:
name:
- description: The header field name
+ description: The header field name.
+ This will be canonicalized upon
+ output, so case-variant names
+ will be understood as the same
+ header.
type: string
value:
description: The header field value
@@ -4807,8 +4937,7 @@ spec:
type: integer
grpc:
description: GRPC specifies an action involving
- a GRPC port. This is a beta field and requires
- enabling GRPCContainerProbe feature gate.
+ a GRPC port.
properties:
port:
description: Port number of the gRPC service.
@@ -4842,7 +4971,10 @@ spec:
header to be used in HTTP probes
properties:
name:
- description: The header field name
+ description: The header field name.
+ This will be canonicalized upon output,
+ so case-variant names will be understood
+ as the same header.
type: string
value:
description: The header field value
@@ -5024,8 +5156,7 @@ spec:
type: integer
grpc:
description: GRPC specifies an action involving
- a GRPC port. This is a beta field and requires
- enabling GRPCContainerProbe feature gate.
+ a GRPC port.
properties:
port:
description: Port number of the gRPC service.
@@ -5059,7 +5190,10 @@ spec:
header to be used in HTTP probes
properties:
name:
- description: The header field name
+ description: The header field name.
+ This will be canonicalized upon output,
+ so case-variant names will be understood
+ as the same header.
type: string
value:
description: The header field value
@@ -5154,6 +5288,28 @@ spec:
format: int32
type: integer
type: object
+ resizePolicy:
+ description: Resources resize policy for the container.
+ items:
+ description: ContainerResizePolicy represents resource
+ resize policy for the container.
+ properties:
+ resourceName:
+ description: 'Name of the resource to which
+ this resource resize policy applies. Supported
+ values: cpu, memory.'
+ type: string
+ restartPolicy:
+ description: Restart policy to apply when specified
+ resource is resized. If not specified, it
+ defaults to NotRequired.
+ type: string
+ required:
+ - resourceName
+ - restartPolicy
+ type: object
+ type: array
+ x-kubernetes-list-type: atomic
resources:
description: 'Compute Resources required by this container.
Cannot be updated. More info: https://kubernetes.io/docs/concepts/configuration/manage-resources-containers/'
@@ -5203,10 +5359,33 @@ spec:
of compute resources required. If Requests is
omitted for a container, it defaults to Limits
if that is explicitly specified, otherwise to
- an implementation-defined value. More info:
- https://kubernetes.io/docs/concepts/configuration/manage-resources-containers/'
+ an implementation-defined value. Requests cannot
+ exceed Limits. More info: https://kubernetes.io/docs/concepts/configuration/manage-resources-containers/'
type: object
type: object
+ restartPolicy:
+ description: 'RestartPolicy defines the restart behavior
+ of individual containers in a pod. This field may
+ only be set for init containers, and the only allowed
+ value is "Always". For non-init containers or when
+ this field is not specified, the restart behavior
+ is defined by the Pod''s restart policy and the
+ container type. Setting the RestartPolicy as "Always"
+ for the init container will have the following effect:
+ this init container will be continually restarted
+ on exit until all regular containers have terminated.
+ Once all regular containers have completed, all
+ init containers with restartPolicy "Always" will
+ be shut down. This lifecycle differs from normal
+ init containers and is often referred to as a "sidecar"
+ container. Although this init container still starts
+ in the init container sequence, it does not wait
+ for the container to complete before proceeding
+ to the next init container. Instead, the next init
+ container starts immediately after this init container
+ is started, or after any startupProbe has successfully
+ completed.'
+ type: string
securityContext:
description: 'SecurityContext defines the security
options the container should be run with. If set,
@@ -5342,8 +5521,9 @@ spec:
be used. The profile must be preconfigured
on the node to work. Must be a descending
path, relative to the kubelet's configured
- seccomp profile location. Must only be set
- if type is "Localhost".
+ seccomp profile location. Must be set if
+ type is "Localhost". Must NOT be set for
+ any other type.
type: string
type:
description: "type indicates which kind of
@@ -5380,17 +5560,12 @@ spec:
hostProcess:
description: HostProcess determines if a container
should be run as a 'Host Process' container.
- This field is alpha-level and will only
- be honored by components that enable the
- WindowsHostProcessContainers feature flag.
- Setting this field without the feature flag
- will result in errors when validating the
- Pod. All of a Pod's containers must have
- the same effective HostProcess value (it
- is not allowed to have a mix of HostProcess
- containers and non-HostProcess containers). In
- addition, if HostProcess is true then HostNetwork
- must also be set to true.
+ All of a Pod's containers must have the
+ same effective HostProcess value (it is
+ not allowed to have a mix of HostProcess
+ containers and non-HostProcess containers).
+ In addition, if HostProcess is true then
+ HostNetwork must also be set to true.
type: boolean
runAsUserName:
description: The UserName in Windows to run
@@ -5441,8 +5616,7 @@ spec:
type: integer
grpc:
description: GRPC specifies an action involving
- a GRPC port. This is a beta field and requires
- enabling GRPCContainerProbe feature gate.
+ a GRPC port.
properties:
port:
description: Port number of the gRPC service.
@@ -5476,7 +5650,10 @@ spec:
header to be used in HTTP probes
properties:
name:
- description: The header field name
+ description: The header field name.
+ This will be canonicalized upon output,
+ so case-variant names will be understood
+ as the same header.
type: string
value:
description: The header field value
@@ -5832,18 +6009,13 @@ spec:
will be used to create a new ResourceClaim,
which will be bound to this pod. When this pod
is deleted, the ResourceClaim will also be deleted.
- The name of the ResourceClaim will be -, where is the PodResourceClaim.Name.
- Pod validation will reject the pod if the concatenated
- name is not valid for a ResourceClaim (e.g.
- too long). \n An existing ResourceClaim with
- that name that is not owned by the pod will
- not be used for the pod to avoid using an unrelated
- resource by mistake. Scheduling and pod startup
- are then blocked until the unrelated ResourceClaim
- is removed. \n This field is immutable and no
- changes will be made to the corresponding ResourceClaim
- by the control plane after creating the ResourceClaim."
+ The pod name and resource name, along with a
+ generated component, will be used to form a
+ unique name for the ResourceClaim, which will
+ be recorded in pod.status.resourceClaimStatuses.
+ \n This field is immutable and no changes will
+ be made to the corresponding ResourceClaim by
+ the control plane after creating the ResourceClaim."
type: string
type: object
required:
@@ -5855,8 +6027,9 @@ spec:
x-kubernetes-list-type: map
restartPolicy:
description: 'Restart policy for all containers within the
- pod. One of Always, OnFailure, Never. Default to Always.
- More info: https://kubernetes.io/docs/concepts/workloads/pods/pod-lifecycle/#restart-policy'
+ pod. One of Always, OnFailure, Never. In some contexts,
+ only a subset of those values may be permitted. Default
+ to Always. More info: https://kubernetes.io/docs/concepts/workloads/pods/pod-lifecycle/#restart-policy'
type: string
runtimeClassName:
description: 'RuntimeClassName refers to a RuntimeClass
@@ -5874,10 +6047,13 @@ spec:
type: string
schedulingGates:
description: "SchedulingGates is an opaque list of values
- that if specified will block scheduling the pod. More
- info: https://git.k8s.io/enhancements/keps/sig-scheduling/3521-pod-scheduling-readiness.
- \n This is an alpha-level feature enabled by PodSchedulingReadiness
- feature gate."
+ that if specified will block scheduling the pod. If schedulingGates
+ is not empty, the pod will stay in the SchedulingGated
+ state and the scheduler will not attempt to schedule the
+ pod. \n SchedulingGates can only be set at pod creation
+ time, and be removed only afterwards. \n This is a beta
+ feature enabled by the PodSchedulingReadiness feature
+ gate."
items:
description: PodSchedulingGate is associated to a Pod
to guard its scheduling.
@@ -5988,7 +6164,8 @@ spec:
The profile must be preconfigured on the node
to work. Must be a descending path, relative to
the kubelet's configured seccomp profile location.
- Must only be set if type is "Localhost".
+ Must be set if type is "Localhost". Must NOT be
+ set for any other type.
type: string
type:
description: "type indicates which kind of seccomp
@@ -6058,15 +6235,11 @@ spec:
type: string
hostProcess:
description: HostProcess determines if a container
- should be run as a 'Host Process' container. This
- field is alpha-level and will only be honored
- by components that enable the WindowsHostProcessContainers
- feature flag. Setting this field without the feature
- flag will result in errors when validating the
- Pod. All of a Pod's containers must have the same
- effective HostProcess value (it is not allowed
- to have a mix of HostProcess containers and non-HostProcess
- containers). In addition, if HostProcess is true
+ should be run as a 'Host Process' container. All
+ of a Pod's containers must have the same effective
+ HostProcess value (it is not allowed to have a
+ mix of HostProcess containers and non-HostProcess
+ containers). In addition, if HostProcess is true
then HostNetwork must also be set to true.
type: boolean
runAsUserName:
@@ -6228,16 +6401,22 @@ spec:
The requirements are ANDed.
type: object
type: object
+ x-kubernetes-map-type: atomic
matchLabelKeys:
- description: MatchLabelKeys is a set of pod label
+ description: "MatchLabelKeys is a set of pod label
keys to select the pods over which spreading will
be calculated. The keys are used to lookup values
from the incoming pod labels, those key-value labels
are ANDed with labelSelector to select the group
of existing pods over which spreading will be calculated
- for the incoming pod. Keys that don't exist in the
- incoming pod labels will be ignored. A null or empty
- list means only match against labelSelector.
+ for the incoming pod. The same key is forbidden
+ to exist in both MatchLabelKeys and LabelSelector.
+ MatchLabelKeys cannot be set when LabelSelector
+ isn't set. Keys that don't exist in the incoming
+ pod labels will be ignored. A null or empty list
+ means only match against labelSelector. \n This
+ is a beta field and requires the MatchLabelKeysInPodTopologySpread
+ feature gate to be enabled (enabled by default)."
items:
type: string
type: array
@@ -6506,6 +6685,7 @@ spec:
kind, uid?'
type: string
type: object
+ x-kubernetes-map-type: atomic
user:
description: 'user is optional: User is the rados
user name, default is admin More info: https://examples.k8s.io/volumes/cephfs/README.md#how-to-use-it'
@@ -6542,6 +6722,7 @@ spec:
kind, uid?'
type: string
type: object
+ x-kubernetes-map-type: atomic
volumeID:
description: 'volumeID used to identify the volume
in cinder. More info: https://examples.k8s.io/mysql-cinder-pd/README.md'
@@ -6623,6 +6804,7 @@ spec:
or its keys must be defined
type: boolean
type: object
+ x-kubernetes-map-type: atomic
csi:
description: csi (Container Storage Interface) represents
ephemeral storage that is handled by certain external
@@ -6657,6 +6839,7 @@ spec:
kind, uid?'
type: string
type: object
+ x-kubernetes-map-type: atomic
readOnly:
description: readOnly specifies a read-only configuration
for the volume. Defaults to false (read/write).
@@ -6716,6 +6899,7 @@ spec:
required:
- fieldPath
type: object
+ x-kubernetes-map-type: atomic
mode:
description: 'Optional: mode bits used to
set permissions on this file, must be
@@ -6764,6 +6948,7 @@ spec:
required:
- resource
type: object
+ x-kubernetes-map-type: atomic
required:
- path
type: object
@@ -6792,7 +6977,7 @@ spec:
specified here and the sum of memory limits
of all containers in a pod. The default is nil
which means that the limit is undefined. More
- info: http://kubernetes.io/docs/user-guide/volumes#emptydir'
+ info: https://kubernetes.io/docs/concepts/storage/volumes#emptydir'
pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$
x-kubernetes-int-or-string: true
type: object
@@ -6918,6 +7103,7 @@ spec:
- kind
- name
type: object
+ x-kubernetes-map-type: atomic
dataSourceRef:
description: 'dataSourceRef specifies
the object from which to populate the
@@ -7052,7 +7238,8 @@ spec:
for a container, it defaults to
Limits if that is explicitly specified,
otherwise to an implementation-defined
- value. More info: https://kubernetes.io/docs/concepts/configuration/manage-resources-containers/'
+ value. Requests cannot exceed Limits.
+ More info: https://kubernetes.io/docs/concepts/configuration/manage-resources-containers/'
type: object
type: object
selector:
@@ -7112,6 +7299,7 @@ spec:
ANDed.
type: object
type: object
+ x-kubernetes-map-type: atomic
storageClassName:
description: 'storageClassName is the
name of the StorageClass required by
@@ -7213,6 +7401,7 @@ spec:
kind, uid?'
type: string
type: object
+ x-kubernetes-map-type: atomic
required:
- driver
type: object
@@ -7406,6 +7595,7 @@ spec:
kind, uid?'
type: string
type: object
+ x-kubernetes-map-type: atomic
targetPortal:
description: targetPortal is iSCSI Target Portal.
The Portal is either an IP or ip_addr:port if
@@ -7593,6 +7783,7 @@ spec:
defined
type: boolean
type: object
+ x-kubernetes-map-type: atomic
downwardAPI:
description: downwardAPI information about
the downwardAPI data to project
@@ -7625,6 +7816,7 @@ spec:
required:
- fieldPath
type: object
+ x-kubernetes-map-type: atomic
mode:
description: 'Optional: mode bits
used to set permissions on this
@@ -7680,6 +7872,7 @@ spec:
required:
- resource
type: object
+ x-kubernetes-map-type: atomic
required:
- path
type: object
@@ -7755,6 +7948,7 @@ spec:
be defined
type: boolean
type: object
+ x-kubernetes-map-type: atomic
serviceAccountToken:
description: serviceAccountToken is information
about the serviceAccountToken data to
@@ -7882,6 +8076,7 @@ spec:
kind, uid?'
type: string
type: object
+ x-kubernetes-map-type: atomic
user:
description: 'user is the rados user name. Default
is admin. More info: https://examples.k8s.io/volumes/rbd/README.md#how-to-use-it'
@@ -7927,6 +8122,7 @@ spec:
kind, uid?'
type: string
type: object
+ x-kubernetes-map-type: atomic
sslEnabled:
description: sslEnabled Flag enable/disable SSL
communication with Gateway, default false
@@ -8054,6 +8250,7 @@ spec:
kind, uid?'
type: string
type: object
+ x-kubernetes-map-type: atomic
volumeName:
description: volumeName is the human-readable
name of the StorageOS volume. Volume names
@@ -8318,6 +8515,235 @@ spec:
- Parallel
type: string
type: object
+ rsmSpec:
+ description: RSMSpec defines workload related spec of this component.
+ start from KB 0.7.0, RSM(ReplicatedStateMachineSpec) will
+ be the underlying CR which powers all kinds of workload in
+ KB. RSM is an enhanced stateful workload extension dedicated
+ for heavy-state workloads like databases.
+ properties:
+ memberUpdateStrategy:
+ description: 'MemberUpdateStrategy, Members(Pods) update
+ strategy. serial: update Members one by one that guarantee
+ minimum component unavailable time. Learner -> Follower(with
+ AccessMode=none) -> Follower(with AccessMode=readonly)
+ -> Follower(with AccessMode=readWrite) -> Leader bestEffortParallel:
+ update Members in parallel that guarantee minimum component
+ un-writable time. Learner, Follower(minority) in parallel
+ -> Follower(majority) -> Leader, keep majority online
+ all the time. parallel: force parallel'
+ enum:
+ - Serial
+ - BestEffortParallel
+ - Parallel
+ type: string
+ membershipReconfiguration:
+ description: MembershipReconfiguration provides actions
+ to do membership dynamic reconfiguration.
+ properties:
+ logSyncAction:
+ description: LogSyncAction specifies how to trigger
+ the new member to start log syncing previous none-nil
+ action's Image wil be used if not configured
+ properties:
+ command:
+ description: Command will be executed in Container
+ to retrieve or process role info
+ items:
+ type: string
+ type: array
+ image:
+ description: utility image contains command that
+ can be used to retrieve of process role info
+ type: string
+ required:
+ - command
+ type: object
+ memberJoinAction:
+ description: MemberJoinAction specifies how to add member
+ previous none-nil action's Image wil be used if not
+ configured
+ properties:
+ command:
+ description: Command will be executed in Container
+ to retrieve or process role info
+ items:
+ type: string
+ type: array
+ image:
+ description: utility image contains command that
+ can be used to retrieve of process role info
+ type: string
+ required:
+ - command
+ type: object
+ memberLeaveAction:
+ description: MemberLeaveAction specifies how to remove
+ member previous none-nil action's Image wil be used
+ if not configured
+ properties:
+ command:
+ description: Command will be executed in Container
+ to retrieve or process role info
+ items:
+ type: string
+ type: array
+ image:
+ description: utility image contains command that
+ can be used to retrieve of process role info
+ type: string
+ required:
+ - command
+ type: object
+ promoteAction:
+ description: PromoteAction specifies how to tell the
+ cluster that the new member can join voting now previous
+ none-nil action's Image wil be used if not configured
+ properties:
+ command:
+ description: Command will be executed in Container
+ to retrieve or process role info
+ items:
+ type: string
+ type: array
+ image:
+ description: utility image contains command that
+ can be used to retrieve of process role info
+ type: string
+ required:
+ - command
+ type: object
+ switchoverAction:
+ description: SwitchoverAction specifies how to do switchover
+ latest [BusyBox](https://busybox.net/) image will
+ be used if Image not configured
+ properties:
+ command:
+ description: Command will be executed in Container
+ to retrieve or process role info
+ items:
+ type: string
+ type: array
+ image:
+ description: utility image contains command that
+ can be used to retrieve of process role info
+ type: string
+ required:
+ - command
+ type: object
+ type: object
+ roleProbe:
+ description: RoleProbe provides method to probe role.
+ properties:
+ failureThreshold:
+ default: 3
+ description: Minimum consecutive failures for the probe
+ to be considered failed after having succeeded. Defaults
+ to 3. Minimum value is 1.
+ format: int32
+ minimum: 1
+ type: integer
+ initialDelaySeconds:
+ default: 0
+ description: Number of seconds after the container has
+ started before role probe has started.
+ format: int32
+ minimum: 0
+ type: integer
+ periodSeconds:
+ default: 2
+ description: How often (in seconds) to perform the probe.
+ Default to 2 seconds. Minimum value is 1.
+ format: int32
+ minimum: 1
+ type: integer
+ probeActions:
+ description: 'ProbeActions define Actions to be taken
+ in serial. after all actions done, the final output
+ should be a single string of the role name defined
+ in spec.Roles latest [BusyBox](https://busybox.net/)
+ image will be used if Image not configured Environment
+ variables can be used in Command: - v_KB_RSM_LAST_STDOUT
+ stdout from last action, watch ''v_'' prefixed - KB_RSM_USERNAME
+ username part of credential - KB_RSM_PASSWORD password
+ part of credential'
+ items:
+ properties:
+ command:
+ description: Command will be executed in Container
+ to retrieve or process role info
+ items:
+ type: string
+ type: array
+ image:
+ description: utility image contains command that
+ can be used to retrieve of process role info
+ type: string
+ required:
+ - command
+ type: object
+ type: array
+ roleUpdateMechanism:
+ default: None
+ description: RoleUpdateMechanism specifies the way how
+ pod role label being updated.
+ enum:
+ - ReadinessProbeEventUpdate
+ - DirectAPIServerEventUpdate
+ - None
+ type: string
+ successThreshold:
+ default: 1
+ description: Minimum consecutive successes for the probe
+ to be considered successful after having failed. Defaults
+ to 1. Minimum value is 1.
+ format: int32
+ minimum: 1
+ type: integer
+ timeoutSeconds:
+ default: 1
+ description: Number of seconds after which the probe
+ times out. Defaults to 1 second. Minimum value is
+ 1.
+ format: int32
+ minimum: 1
+ type: integer
+ required:
+ - probeActions
+ type: object
+ roles:
+ description: Roles, a list of roles defined in the system.
+ items:
+ properties:
+ accessMode:
+ default: ReadWrite
+ description: AccessMode, what service this member
+ capable.
+ enum:
+ - None
+ - Readonly
+ - ReadWrite
+ type: string
+ canVote:
+ default: true
+ description: CanVote, whether this member has voting
+ rights
+ type: boolean
+ isLeader:
+ default: false
+ description: IsLeader, whether this member is the
+ leader
+ type: boolean
+ name:
+ default: leader
+ description: Name, role name.
+ type: string
+ required:
+ - accessMode
+ - name
+ type: object
+ type: array
+ type: object
scriptSpecs:
description: The scriptSpec field provided by provider, and
finally this configTemplateRefs will be rendered into the
@@ -8698,6 +9124,7 @@ spec:
required:
- key
type: object
+ x-kubernetes-map-type: atomic
fieldRef:
description: 'Selects a field of the pod:
supports metadata.name, metadata.namespace,
@@ -8717,6 +9144,7 @@ spec:
required:
- fieldPath
type: object
+ x-kubernetes-map-type: atomic
resourceFieldRef:
description: 'Selects a resource of the
container: only resources limits and
@@ -8745,6 +9173,7 @@ spec:
required:
- resource
type: object
+ x-kubernetes-map-type: atomic
secretKeyRef:
description: Selects a key of a secret
in the pod's namespace
@@ -8767,6 +9196,7 @@ spec:
required:
- key
type: object
+ x-kubernetes-map-type: atomic
type: object
required:
- name
@@ -8867,6 +9297,7 @@ spec:
required:
- key
type: object
+ x-kubernetes-map-type: atomic
fieldRef:
description: 'Selects a field of the pod:
supports metadata.name, metadata.namespace,
@@ -8886,6 +9317,7 @@ spec:
required:
- fieldPath
type: object
+ x-kubernetes-map-type: atomic
resourceFieldRef:
description: 'Selects a resource of the
container: only resources limits and
@@ -8914,6 +9346,7 @@ spec:
required:
- resource
type: object
+ x-kubernetes-map-type: atomic
secretKeyRef:
description: Selects a key of a secret
in the pod's namespace
@@ -8936,6 +9369,7 @@ spec:
required:
- key
type: object
+ x-kubernetes-map-type: atomic
type: object
required:
- name
@@ -9118,6 +9552,7 @@ spec:
required:
- key
type: object
+ x-kubernetes-map-type: atomic
fieldRef:
description: 'Selects a field of the pod:
supports metadata.name, metadata.namespace,
@@ -9137,6 +9572,7 @@ spec:
required:
- fieldPath
type: object
+ x-kubernetes-map-type: atomic
resourceFieldRef:
description: 'Selects a resource of the container:
only resources limits and requests (limits.cpu,
@@ -9163,6 +9599,7 @@ spec:
required:
- resource
type: object
+ x-kubernetes-map-type: atomic
secretKeyRef:
description: Selects a key of a secret in
the pod's namespace
@@ -9185,6 +9622,7 @@ spec:
required:
- key
type: object
+ x-kubernetes-map-type: atomic
type: object
required:
- name
@@ -9323,10 +9761,11 @@ spec:
- workloadType
type: object
x-kubernetes-validations:
- - message: componentDefs.consensusSpec is required when componentDefs.workloadType
- is Consensus, and forbidden otherwise
+ - message: componentDefs.consensusSpec(deprecated) or componentDefs.rsmSpec(recommended)
+ is required when componentDefs.workloadType is Consensus, and
+ forbidden otherwise
rule: 'has(self.workloadType) && self.workloadType == ''Consensus''
- ? has(self.consensusSpec) : !has(self.consensusSpec)'
+ ? (has(self.consensusSpec) || has(self.rsmSpec)) : !has(self.consensusSpec)'
minItems: 1
type: array
x-kubernetes-list-map-keys:
diff --git a/config/crd/bases/apps.kubeblocks.io_clusters.yaml b/config/crd/bases/apps.kubeblocks.io_clusters.yaml
index 18c2abf550a..d741c62fef4 100644
--- a/config/crd/bases/apps.kubeblocks.io_clusters.yaml
+++ b/config/crd/bases/apps.kubeblocks.io_clusters.yaml
@@ -2,8 +2,7 @@ apiVersion: apiextensions.k8s.io/v1
kind: CustomResourceDefinition
metadata:
annotations:
- controller-gen.kubebuilder.io/version: v0.9.0
- creationTimestamp: null
+ controller-gen.kubebuilder.io/version: v0.12.1
labels:
app.kubernetes.io/name: kubeblocks
name: clusters.apps.kubeblocks.io
@@ -119,15 +118,7 @@ spec:
description: enabled defines whether to enable automated backup.
type: boolean
method:
- allOf:
- - enum:
- - snapshot
- - backupTool
- - enum:
- - snapshot
- - backupTool
- default: snapshot
- description: 'backup method, support: snapshot, backupTool.'
+ description: backup method name to use, that is defined in backupPolicy.
type: string
pitrEnabled:
default: false
@@ -139,11 +130,14 @@ spec:
will use the default backupRepo.
type: string
retentionPeriod:
- default: 1d
- description: retentionPeriod is a time string ending with the
- 'd'|'D'|'h'|'H' character to describe how long the Backup should
- be retained. if not set, will be retained forever.
- pattern: ^\d+[d|D|h|H]$
+ default: 7d
+ description: "retentionPeriod determines a duration up to which
+ the backup should be kept. controller will remove all backups
+ that are older than the RetentionPeriod. For example, RetentionPeriod
+ of `30d` will keep only the backups of last 30 days. Sample
+ duration format: - years: \t2y - months: \t6mo - days: \t\t30d
+ - hours: \t12h - minutes: \t30m You can also combine the above
+ durations. For example: 30d12h30m"
type: string
startingDeadlineMinutes:
description: startingDeadlineMinutes defines the deadline in minutes
@@ -153,8 +147,6 @@ spec:
maximum: 1440
minimum: 0
type: integer
- required:
- - method
type: object
clusterDefinitionRef:
description: Cluster referencing ClusterDefinition name. This is an
@@ -362,8 +354,8 @@ spec:
description: 'Requests describes the minimum amount of compute
resources required. If Requests is omitted for a container,
it defaults to Limits if that is explicitly specified,
- otherwise to an implementation-defined value. More info:
- https://kubernetes.io/docs/concepts/configuration/manage-resources-containers/'
+ otherwise to an implementation-defined value. Requests
+ cannot exceed Limits. More info: https://kubernetes.io/docs/concepts/configuration/manage-resources-containers/'
type: object
type: object
x-kubernetes-preserve-unknown-fields: true
@@ -622,8 +614,8 @@ spec:
of compute resources required. If Requests is
omitted for a container, it defaults to Limits
if that is explicitly specified, otherwise to
- an implementation-defined value. More info:
- https://kubernetes.io/docs/concepts/configuration/manage-resources-containers/'
+ an implementation-defined value. Requests cannot
+ exceed Limits. More info: https://kubernetes.io/docs/concepts/configuration/manage-resources-containers/'
type: object
type: object
x-kubernetes-preserve-unknown-fields: true
@@ -873,6 +865,48 @@ spec:
required:
- leader
type: object
+ membersStatus:
+ description: members' status.
+ items:
+ properties:
+ podName:
+ default: Unknown
+ description: PodName pod name.
+ type: string
+ role:
+ properties:
+ accessMode:
+ default: ReadWrite
+ description: AccessMode, what service this member
+ capable.
+ enum:
+ - None
+ - Readonly
+ - ReadWrite
+ type: string
+ canVote:
+ default: true
+ description: CanVote, whether this member has voting
+ rights
+ type: boolean
+ isLeader:
+ default: false
+ description: IsLeader, whether this member is the
+ leader
+ type: boolean
+ name:
+ default: leader
+ description: Name, role name.
+ type: string
+ required:
+ - accessMode
+ - name
+ type: object
+ required:
+ - podName
+ - role
+ type: object
+ type: array
message:
additionalProperties:
type: string
diff --git a/config/crd/bases/apps.kubeblocks.io_clusterversions.yaml b/config/crd/bases/apps.kubeblocks.io_clusterversions.yaml
index 77302045da9..3e9507ef482 100644
--- a/config/crd/bases/apps.kubeblocks.io_clusterversions.yaml
+++ b/config/crd/bases/apps.kubeblocks.io_clusterversions.yaml
@@ -2,8 +2,7 @@ apiVersion: apiextensions.k8s.io/v1
kind: CustomResourceDefinition
metadata:
annotations:
- controller-gen.kubebuilder.io/version: v0.9.0
- creationTimestamp: null
+ controller-gen.kubebuilder.io/version: v0.12.1
labels:
app.kubernetes.io/name: kubeblocks
name: clusterversions.apps.kubeblocks.io
@@ -230,6 +229,7 @@ spec:
required:
- key
type: object
+ x-kubernetes-map-type: atomic
fieldRef:
description: 'Selects a field of the pod:
supports metadata.name, metadata.namespace,
@@ -249,6 +249,7 @@ spec:
required:
- fieldPath
type: object
+ x-kubernetes-map-type: atomic
resourceFieldRef:
description: 'Selects a resource of the container:
only resources limits and requests (limits.cpu,
@@ -275,6 +276,7 @@ spec:
required:
- resource
type: object
+ x-kubernetes-map-type: atomic
secretKeyRef:
description: Selects a key of a secret in
the pod's namespace
@@ -297,6 +299,7 @@ spec:
required:
- key
type: object
+ x-kubernetes-map-type: atomic
type: object
required:
- name
@@ -370,6 +373,7 @@ spec:
required:
- key
type: object
+ x-kubernetes-map-type: atomic
fieldRef:
description: 'Selects a field of the pod:
supports metadata.name, metadata.namespace,
@@ -389,6 +393,7 @@ spec:
required:
- fieldPath
type: object
+ x-kubernetes-map-type: atomic
resourceFieldRef:
description: 'Selects a resource of the container:
only resources limits and requests (limits.cpu,
@@ -415,6 +420,7 @@ spec:
required:
- resource
type: object
+ x-kubernetes-map-type: atomic
secretKeyRef:
description: Selects a key of a secret in
the pod's namespace
@@ -437,6 +443,7 @@ spec:
required:
- key
type: object
+ x-kubernetes-map-type: atomic
type: object
required:
- name
@@ -545,6 +552,7 @@ spec:
required:
- key
type: object
+ x-kubernetes-map-type: atomic
fieldRef:
description: 'Selects a field of the pod:
supports metadata.name, metadata.namespace,
@@ -564,6 +572,7 @@ spec:
required:
- fieldPath
type: object
+ x-kubernetes-map-type: atomic
resourceFieldRef:
description: 'Selects a resource of the
container: only resources limits and requests
@@ -591,6 +600,7 @@ spec:
required:
- resource
type: object
+ x-kubernetes-map-type: atomic
secretKeyRef:
description: Selects a key of a secret in
the pod's namespace
@@ -613,6 +623,7 @@ spec:
required:
- key
type: object
+ x-kubernetes-map-type: atomic
type: object
required:
- name
@@ -645,6 +656,7 @@ spec:
must be defined
type: boolean
type: object
+ x-kubernetes-map-type: atomic
prefix:
description: An optional identifier to prepend
to each key in the ConfigMap. Must be a C_IDENTIFIER.
@@ -663,6 +675,7 @@ spec:
must be defined
type: boolean
type: object
+ x-kubernetes-map-type: atomic
type: object
type: array
image:
@@ -727,7 +740,11 @@ spec:
custom header to be used in HTTP probes
properties:
name:
- description: The header field name
+ description: The header field name.
+ This will be canonicalized upon
+ output, so case-variant names
+ will be understood as the same
+ header.
type: string
value:
description: The header field value
@@ -835,7 +852,11 @@ spec:
custom header to be used in HTTP probes
properties:
name:
- description: The header field name
+ description: The header field name.
+ This will be canonicalized upon
+ output, so case-variant names
+ will be understood as the same
+ header.
type: string
value:
description: The header field value
@@ -922,8 +943,7 @@ spec:
type: integer
grpc:
description: GRPC specifies an action involving
- a GRPC port. This is a beta field and requires
- enabling GRPCContainerProbe feature gate.
+ a GRPC port.
properties:
port:
description: Port number of the gRPC service.
@@ -957,7 +977,10 @@ spec:
header to be used in HTTP probes
properties:
name:
- description: The header field name
+ description: The header field name.
+ This will be canonicalized upon output,
+ so case-variant names will be understood
+ as the same header.
type: string
value:
description: The header field value
@@ -1139,8 +1162,7 @@ spec:
type: integer
grpc:
description: GRPC specifies an action involving
- a GRPC port. This is a beta field and requires
- enabling GRPCContainerProbe feature gate.
+ a GRPC port.
properties:
port:
description: Port number of the gRPC service.
@@ -1174,7 +1196,10 @@ spec:
header to be used in HTTP probes
properties:
name:
- description: The header field name
+ description: The header field name.
+ This will be canonicalized upon output,
+ so case-variant names will be understood
+ as the same header.
type: string
value:
description: The header field value
@@ -1269,6 +1294,28 @@ spec:
format: int32
type: integer
type: object
+ resizePolicy:
+ description: Resources resize policy for the container.
+ items:
+ description: ContainerResizePolicy represents resource
+ resize policy for the container.
+ properties:
+ resourceName:
+ description: 'Name of the resource to which
+ this resource resize policy applies. Supported
+ values: cpu, memory.'
+ type: string
+ restartPolicy:
+ description: Restart policy to apply when specified
+ resource is resized. If not specified, it
+ defaults to NotRequired.
+ type: string
+ required:
+ - resourceName
+ - restartPolicy
+ type: object
+ type: array
+ x-kubernetes-list-type: atomic
resources:
description: 'Compute Resources required by this container.
Cannot be updated. More info: https://kubernetes.io/docs/concepts/configuration/manage-resources-containers/'
@@ -1318,10 +1365,33 @@ spec:
of compute resources required. If Requests is
omitted for a container, it defaults to Limits
if that is explicitly specified, otherwise to
- an implementation-defined value. More info:
- https://kubernetes.io/docs/concepts/configuration/manage-resources-containers/'
+ an implementation-defined value. Requests cannot
+ exceed Limits. More info: https://kubernetes.io/docs/concepts/configuration/manage-resources-containers/'
type: object
type: object
+ restartPolicy:
+ description: 'RestartPolicy defines the restart behavior
+ of individual containers in a pod. This field may
+ only be set for init containers, and the only allowed
+ value is "Always". For non-init containers or when
+ this field is not specified, the restart behavior
+ is defined by the Pod''s restart policy and the
+ container type. Setting the RestartPolicy as "Always"
+ for the init container will have the following effect:
+ this init container will be continually restarted
+ on exit until all regular containers have terminated.
+ Once all regular containers have completed, all
+ init containers with restartPolicy "Always" will
+ be shut down. This lifecycle differs from normal
+ init containers and is often referred to as a "sidecar"
+ container. Although this init container still starts
+ in the init container sequence, it does not wait
+ for the container to complete before proceeding
+ to the next init container. Instead, the next init
+ container starts immediately after this init container
+ is started, or after any startupProbe has successfully
+ completed.'
+ type: string
securityContext:
description: 'SecurityContext defines the security
options the container should be run with. If set,
@@ -1457,8 +1527,9 @@ spec:
be used. The profile must be preconfigured
on the node to work. Must be a descending
path, relative to the kubelet's configured
- seccomp profile location. Must only be set
- if type is "Localhost".
+ seccomp profile location. Must be set if
+ type is "Localhost". Must NOT be set for
+ any other type.
type: string
type:
description: "type indicates which kind of
@@ -1495,17 +1566,12 @@ spec:
hostProcess:
description: HostProcess determines if a container
should be run as a 'Host Process' container.
- This field is alpha-level and will only
- be honored by components that enable the
- WindowsHostProcessContainers feature flag.
- Setting this field without the feature flag
- will result in errors when validating the
- Pod. All of a Pod's containers must have
- the same effective HostProcess value (it
- is not allowed to have a mix of HostProcess
- containers and non-HostProcess containers). In
- addition, if HostProcess is true then HostNetwork
- must also be set to true.
+ All of a Pod's containers must have the
+ same effective HostProcess value (it is
+ not allowed to have a mix of HostProcess
+ containers and non-HostProcess containers).
+ In addition, if HostProcess is true then
+ HostNetwork must also be set to true.
type: boolean
runAsUserName:
description: The UserName in Windows to run
@@ -1556,8 +1622,7 @@ spec:
type: integer
grpc:
description: GRPC specifies an action involving
- a GRPC port. This is a beta field and requires
- enabling GRPCContainerProbe feature gate.
+ a GRPC port.
properties:
port:
description: Port number of the gRPC service.
@@ -1591,7 +1656,10 @@ spec:
header to be used in HTTP probes
properties:
name:
- description: The header field name
+ description: The header field name.
+ This will be canonicalized upon output,
+ so case-variant names will be understood
+ as the same header.
type: string
value:
description: The header field value
@@ -1900,6 +1968,7 @@ spec:
required:
- key
type: object
+ x-kubernetes-map-type: atomic
fieldRef:
description: 'Selects a field of the pod:
supports metadata.name, metadata.namespace,
@@ -1919,6 +1988,7 @@ spec:
required:
- fieldPath
type: object
+ x-kubernetes-map-type: atomic
resourceFieldRef:
description: 'Selects a resource of the
container: only resources limits and requests
@@ -1946,6 +2016,7 @@ spec:
required:
- resource
type: object
+ x-kubernetes-map-type: atomic
secretKeyRef:
description: Selects a key of a secret in
the pod's namespace
@@ -1968,6 +2039,7 @@ spec:
required:
- key
type: object
+ x-kubernetes-map-type: atomic
type: object
required:
- name
@@ -2000,6 +2072,7 @@ spec:
must be defined
type: boolean
type: object
+ x-kubernetes-map-type: atomic
prefix:
description: An optional identifier to prepend
to each key in the ConfigMap. Must be a C_IDENTIFIER.
@@ -2018,6 +2091,7 @@ spec:
must be defined
type: boolean
type: object
+ x-kubernetes-map-type: atomic
type: object
type: array
image:
@@ -2082,7 +2156,11 @@ spec:
custom header to be used in HTTP probes
properties:
name:
- description: The header field name
+ description: The header field name.
+ This will be canonicalized upon
+ output, so case-variant names
+ will be understood as the same
+ header.
type: string
value:
description: The header field value
@@ -2190,7 +2268,11 @@ spec:
custom header to be used in HTTP probes
properties:
name:
- description: The header field name
+ description: The header field name.
+ This will be canonicalized upon
+ output, so case-variant names
+ will be understood as the same
+ header.
type: string
value:
description: The header field value
@@ -2277,8 +2359,7 @@ spec:
type: integer
grpc:
description: GRPC specifies an action involving
- a GRPC port. This is a beta field and requires
- enabling GRPCContainerProbe feature gate.
+ a GRPC port.
properties:
port:
description: Port number of the gRPC service.
@@ -2312,7 +2393,10 @@ spec:
header to be used in HTTP probes
properties:
name:
- description: The header field name
+ description: The header field name.
+ This will be canonicalized upon output,
+ so case-variant names will be understood
+ as the same header.
type: string
value:
description: The header field value
@@ -2494,8 +2578,7 @@ spec:
type: integer
grpc:
description: GRPC specifies an action involving
- a GRPC port. This is a beta field and requires
- enabling GRPCContainerProbe feature gate.
+ a GRPC port.
properties:
port:
description: Port number of the gRPC service.
@@ -2529,7 +2612,10 @@ spec:
header to be used in HTTP probes
properties:
name:
- description: The header field name
+ description: The header field name.
+ This will be canonicalized upon output,
+ so case-variant names will be understood
+ as the same header.
type: string
value:
description: The header field value
@@ -2624,6 +2710,28 @@ spec:
format: int32
type: integer
type: object
+ resizePolicy:
+ description: Resources resize policy for the container.
+ items:
+ description: ContainerResizePolicy represents resource
+ resize policy for the container.
+ properties:
+ resourceName:
+ description: 'Name of the resource to which
+ this resource resize policy applies. Supported
+ values: cpu, memory.'
+ type: string
+ restartPolicy:
+ description: Restart policy to apply when specified
+ resource is resized. If not specified, it
+ defaults to NotRequired.
+ type: string
+ required:
+ - resourceName
+ - restartPolicy
+ type: object
+ type: array
+ x-kubernetes-list-type: atomic
resources:
description: 'Compute Resources required by this container.
Cannot be updated. More info: https://kubernetes.io/docs/concepts/configuration/manage-resources-containers/'
@@ -2673,10 +2781,33 @@ spec:
of compute resources required. If Requests is
omitted for a container, it defaults to Limits
if that is explicitly specified, otherwise to
- an implementation-defined value. More info:
- https://kubernetes.io/docs/concepts/configuration/manage-resources-containers/'
+ an implementation-defined value. Requests cannot
+ exceed Limits. More info: https://kubernetes.io/docs/concepts/configuration/manage-resources-containers/'
type: object
type: object
+ restartPolicy:
+ description: 'RestartPolicy defines the restart behavior
+ of individual containers in a pod. This field may
+ only be set for init containers, and the only allowed
+ value is "Always". For non-init containers or when
+ this field is not specified, the restart behavior
+ is defined by the Pod''s restart policy and the
+ container type. Setting the RestartPolicy as "Always"
+ for the init container will have the following effect:
+ this init container will be continually restarted
+ on exit until all regular containers have terminated.
+ Once all regular containers have completed, all
+ init containers with restartPolicy "Always" will
+ be shut down. This lifecycle differs from normal
+ init containers and is often referred to as a "sidecar"
+ container. Although this init container still starts
+ in the init container sequence, it does not wait
+ for the container to complete before proceeding
+ to the next init container. Instead, the next init
+ container starts immediately after this init container
+ is started, or after any startupProbe has successfully
+ completed.'
+ type: string
securityContext:
description: 'SecurityContext defines the security
options the container should be run with. If set,
@@ -2812,8 +2943,9 @@ spec:
be used. The profile must be preconfigured
on the node to work. Must be a descending
path, relative to the kubelet's configured
- seccomp profile location. Must only be set
- if type is "Localhost".
+ seccomp profile location. Must be set if
+ type is "Localhost". Must NOT be set for
+ any other type.
type: string
type:
description: "type indicates which kind of
@@ -2850,17 +2982,12 @@ spec:
hostProcess:
description: HostProcess determines if a container
should be run as a 'Host Process' container.
- This field is alpha-level and will only
- be honored by components that enable the
- WindowsHostProcessContainers feature flag.
- Setting this field without the feature flag
- will result in errors when validating the
- Pod. All of a Pod's containers must have
- the same effective HostProcess value (it
- is not allowed to have a mix of HostProcess
- containers and non-HostProcess containers). In
- addition, if HostProcess is true then HostNetwork
- must also be set to true.
+ All of a Pod's containers must have the
+ same effective HostProcess value (it is
+ not allowed to have a mix of HostProcess
+ containers and non-HostProcess containers).
+ In addition, if HostProcess is true then
+ HostNetwork must also be set to true.
type: boolean
runAsUserName:
description: The UserName in Windows to run
@@ -2911,8 +3038,7 @@ spec:
type: integer
grpc:
description: GRPC specifies an action involving
- a GRPC port. This is a beta field and requires
- enabling GRPCContainerProbe feature gate.
+ a GRPC port.
properties:
port:
description: Port number of the gRPC service.
@@ -2946,7 +3072,10 @@ spec:
header to be used in HTTP probes
properties:
name:
- description: The header field name
+ description: The header field name.
+ This will be canonicalized upon output,
+ so case-variant names will be understood
+ as the same header.
type: string
value:
description: The header field value
diff --git a/config/crd/bases/apps.kubeblocks.io_componentclassdefinitions.yaml b/config/crd/bases/apps.kubeblocks.io_componentclassdefinitions.yaml
index 9827863faa8..84569625c77 100644
--- a/config/crd/bases/apps.kubeblocks.io_componentclassdefinitions.yaml
+++ b/config/crd/bases/apps.kubeblocks.io_componentclassdefinitions.yaml
@@ -2,8 +2,7 @@ apiVersion: apiextensions.k8s.io/v1
kind: CustomResourceDefinition
metadata:
annotations:
- controller-gen.kubebuilder.io/version: v0.9.0
- creationTimestamp: null
+ controller-gen.kubebuilder.io/version: v0.12.1
labels:
app.kubernetes.io/name: kubeblocks
name: componentclassdefinitions.apps.kubeblocks.io
diff --git a/config/crd/bases/apps.kubeblocks.io_componentresourceconstraints.yaml b/config/crd/bases/apps.kubeblocks.io_componentresourceconstraints.yaml
index d96a325fe7a..0854049bfa6 100644
--- a/config/crd/bases/apps.kubeblocks.io_componentresourceconstraints.yaml
+++ b/config/crd/bases/apps.kubeblocks.io_componentresourceconstraints.yaml
@@ -2,8 +2,7 @@ apiVersion: apiextensions.k8s.io/v1
kind: CustomResourceDefinition
metadata:
annotations:
- controller-gen.kubebuilder.io/version: v0.9.0
- creationTimestamp: null
+ controller-gen.kubebuilder.io/version: v0.12.1
labels:
app.kubernetes.io/name: kubeblocks
name: componentresourceconstraints.apps.kubeblocks.io
diff --git a/config/crd/bases/apps.kubeblocks.io_configconstraints.yaml b/config/crd/bases/apps.kubeblocks.io_configconstraints.yaml
index 1d9f4c48175..e705f33b453 100644
--- a/config/crd/bases/apps.kubeblocks.io_configconstraints.yaml
+++ b/config/crd/bases/apps.kubeblocks.io_configconstraints.yaml
@@ -2,8 +2,7 @@ apiVersion: apiextensions.k8s.io/v1
kind: CustomResourceDefinition
metadata:
annotations:
- controller-gen.kubebuilder.io/version: v0.9.0
- creationTimestamp: null
+ controller-gen.kubebuilder.io/version: v0.12.1
labels:
app.kubernetes.io/name: kubeblocks
name: configconstraints.apps.kubeblocks.io
@@ -95,6 +94,7 @@ spec:
required:
- fieldPath
type: object
+ x-kubernetes-map-type: atomic
mode:
description: 'Optional: mode bits used to set permissions
on this file, must be an octal value between 0000 and
@@ -135,6 +135,7 @@ spec:
required:
- resource
type: object
+ x-kubernetes-map-type: atomic
required:
- path
type: object
@@ -371,6 +372,7 @@ spec:
are ANDed.
type: object
type: object
+ x-kubernetes-map-type: atomic
staticParameters:
description: staticParameters, list of StaticParameter, modifications
of them trigger a process restart.
diff --git a/config/crd/bases/apps.kubeblocks.io_configurations.yaml b/config/crd/bases/apps.kubeblocks.io_configurations.yaml
index 6cc4adc83b3..7dbc26db996 100644
--- a/config/crd/bases/apps.kubeblocks.io_configurations.yaml
+++ b/config/crd/bases/apps.kubeblocks.io_configurations.yaml
@@ -2,8 +2,7 @@ apiVersion: apiextensions.k8s.io/v1
kind: CustomResourceDefinition
metadata:
annotations:
- controller-gen.kubebuilder.io/version: v0.9.0
- creationTimestamp: null
+ controller-gen.kubebuilder.io/version: v0.12.1
labels:
app.kubernetes.io/name: kubeblocks
name: configurations.apps.kubeblocks.io
@@ -36,21 +35,12 @@ spec:
spec:
description: ConfigurationSpec defines the desired state of Configuration
properties:
- clusterDefRef:
- description: clusterDefRef referencing ClusterDefinition name. This
- is an immutable attribute.
- pattern: ^[a-z0-9]([a-z0-9\.\-]*[a-z0-9])?$
- type: string
clusterRef:
description: clusterRef references Cluster name.
type: string
x-kubernetes-validations:
- message: forbidden to update spec.clusterRef
rule: self == oldSelf
- clusterVerRef:
- description: clusterVerRef referencing ClusterVersion name.
- pattern: ^[a-z0-9]([a-z0-9\.\-]*[a-z0-9])?$
- type: string
componentName:
description: componentName is cluster component name.
type: string
@@ -79,6 +69,104 @@ spec:
description: configFileParams is used to set the parameters
to be updated.
type: object
+ configSpec:
+ description: configSpec is used to set the configuration template.
+ properties:
+ asEnvFrom:
+ description: 'asEnvFrom is optional: the list of containers
+ will be injected into EnvFrom.'
+ items:
+ type: string
+ type: array
+ x-kubernetes-list-type: set
+ constraintRef:
+ description: Specify the name of the referenced the configuration
+ constraints object.
+ maxLength: 63
+ pattern: ^[a-z0-9]([a-z0-9\.\-]*[a-z0-9])?$
+ type: string
+ defaultMode:
+ description: 'defaultMode is optional: mode bits used to
+ set permissions on created files by default. Must be an
+ octal value between 0000 and 0777 or a decimal value between
+ 0 and 511. YAML accepts both octal and decimal values,
+ JSON requires decimal values for mode bits. Defaults to
+ 0644. Directories within the path are not affected by
+ this setting. This might be in conflict with other options
+ that affect the file mode, like fsGroup, and the result
+ can be other mode bits set.'
+ format: int32
+ type: integer
+ keys:
+ description: Specify a list of keys. If empty, ConfigConstraint
+ takes effect for all keys in configmap.
+ items:
+ type: string
+ type: array
+ x-kubernetes-list-type: set
+ legacyRenderedConfigSpec:
+ description: 'lazyRenderedConfigSpec is optional: specify
+ the secondary rendered config spec.'
+ properties:
+ namespace:
+ default: default
+ description: Specify the namespace of the referenced
+ the configuration template ConfigMap object. An empty
+ namespace is equivalent to the "default" namespace.
+ maxLength: 63
+ pattern: ^[a-z0-9]([a-z0-9\-]*[a-z0-9])?$
+ type: string
+ policy:
+ default: none
+ description: policy defines how to merge external imported
+ templates into component templates.
+ enum:
+ - patch
+ - replace
+ - none
+ type: string
+ templateRef:
+ description: Specify the name of the referenced the
+ configuration template ConfigMap object.
+ maxLength: 63
+ pattern: ^[a-z0-9]([a-z0-9\.\-]*[a-z0-9])?$
+ type: string
+ required:
+ - templateRef
+ type: object
+ name:
+ description: Specify the name of configuration template.
+ maxLength: 63
+ pattern: ^[a-z0-9]([a-z0-9\.\-]*[a-z0-9])?$
+ type: string
+ namespace:
+ default: default
+ description: Specify the namespace of the referenced the
+ configuration template ConfigMap object. An empty namespace
+ is equivalent to the "default" namespace.
+ maxLength: 63
+ pattern: ^[a-z0-9]([a-z0-9\-]*[a-z0-9])?$
+ type: string
+ templateRef:
+ description: Specify the name of the referenced the configuration
+ template ConfigMap object.
+ maxLength: 63
+ pattern: ^[a-z0-9]([a-z0-9\.\-]*[a-z0-9])?$
+ type: string
+ volumeName:
+ description: volumeName is the volume name of PodTemplate,
+ which the configuration file produced through the configuration
+ template will be mounted to the corresponding volume.
+ Must be a DNS_LABEL name. The volume name must be defined
+ in podSpec.containers[*].volumeMounts.
+ maxLength: 63
+ pattern: ^[a-z]([a-z0-9\-]*[a-z0-9])?$
+ type: string
+ required:
+ - name
+ - templateRef
+ - volumeName
+ type: object
importTemplateRef:
description: Specify the configuration template.
properties:
@@ -124,7 +212,6 @@ spec:
- name
x-kubernetes-list-type: map
required:
- - clusterDefRef
- clusterRef
- componentName
type: object
@@ -223,6 +310,7 @@ spec:
phase:
description: phase is status of configurationItem.
enum:
+ - Creating
- Init
- Running
- Pending
diff --git a/config/crd/bases/apps.kubeblocks.io_opsrequests.yaml b/config/crd/bases/apps.kubeblocks.io_opsrequests.yaml
index 140b9cbb572..16bf1252b31 100644
--- a/config/crd/bases/apps.kubeblocks.io_opsrequests.yaml
+++ b/config/crd/bases/apps.kubeblocks.io_opsrequests.yaml
@@ -2,8 +2,7 @@ apiVersion: apiextensions.k8s.io/v1
kind: CustomResourceDefinition
metadata:
annotations:
- controller-gen.kubebuilder.io/version: v0.9.0
- creationTimestamp: null
+ controller-gen.kubebuilder.io/version: v0.12.1
labels:
app.kubernetes.io/name: kubeblocks
name: opsrequests.apps.kubeblocks.io
@@ -64,17 +63,15 @@ spec:
backupSpec:
description: backupSpec defines how to backup the cluster.
properties:
+ backupMethod:
+ description: Backup method name that is defined in backupPolicy.
+ type: string
backupName:
description: backupName is the name of the backup.
type: string
backupPolicyName:
description: Which backupPolicy is applied to perform this backup
type: string
- backupType:
- default: datafile
- description: Backup Type. datafile or logfile or snapshot. If
- not set, datafile is the default type.
- type: string
parentBackupName:
description: if backupType is incremental, parentBackupName is
required.
@@ -363,6 +360,7 @@ spec:
required:
- key
type: object
+ x-kubernetes-map-type: atomic
type: array
x-kubernetes-validations:
- message: forbidden to update spec.scriptSpec.scriptFrom.configMapRef
@@ -387,6 +385,7 @@ spec:
required:
- key
type: object
+ x-kubernetes-map-type: atomic
type: array
x-kubernetes-validations:
- message: forbidden to update spec.scriptSpec.scriptFrom.secretRef
@@ -418,6 +417,59 @@ spec:
required:
- name
type: object
+ selector:
+ description: KubeBlocks, by default, will execute the script on
+ the primary pod, with role=leader. There are some exceptions,
+ such as Redis, which does not synchronize accounts info between
+ primary and secondary. In this case, we need to execute the
+ script on all pods, matching the selector. selector indicates
+ the components on which the script is executed.
+ properties:
+ matchExpressions:
+ description: matchExpressions is a list of label selector
+ requirements. The requirements are ANDed.
+ items:
+ description: A label selector requirement is a selector
+ that contains values, a key, and an operator that relates
+ the key and values.
+ properties:
+ key:
+ description: key is the label key that the selector
+ applies to.
+ type: string
+ operator:
+ description: operator represents a key's relationship
+ to a set of values. Valid operators are In, NotIn,
+ Exists and DoesNotExist.
+ type: string
+ values:
+ description: values is an array of string values. If
+ the operator is In or NotIn, the values array must
+ be non-empty. If the operator is Exists or DoesNotExist,
+ the values array must be empty. This array is replaced
+ during a strategic merge patch.
+ items:
+ type: string
+ type: array
+ required:
+ - key
+ - operator
+ type: object
+ type: array
+ matchLabels:
+ additionalProperties:
+ type: string
+ description: matchLabels is a map of {key,value} pairs. A
+ single {key,value} in the matchLabels map is equivalent
+ to an element of matchExpressions, whose key field is "key",
+ the operator is "In", and the values array contains only
+ "value". The requirements are ANDed.
+ type: object
+ type: object
+ x-kubernetes-map-type: atomic
+ x-kubernetes-validations:
+ - message: forbidden to update spec.scriptSpec.script.selector
+ rule: self == oldSelf
required:
- componentName
type: object
@@ -564,7 +616,8 @@ spec:
description: 'Requests describes the minimum amount of compute
resources required. If Requests is omitted for a container,
it defaults to Limits if that is explicitly specified, otherwise
- to an implementation-defined value. More info: https://kubernetes.io/docs/concepts/configuration/manage-resources-containers/'
+ to an implementation-defined value. Requests cannot exceed
+ Limits. More info: https://kubernetes.io/docs/concepts/configuration/manage-resources-containers/'
type: object
required:
- componentName
@@ -867,8 +920,8 @@ spec:
description: 'Requests describes the minimum amount of compute
resources required. If Requests is omitted for a container,
it defaults to Limits if that is explicitly specified,
- otherwise to an implementation-defined value. More info:
- https://kubernetes.io/docs/concepts/configuration/manage-resources-containers/'
+ otherwise to an implementation-defined value. Requests
+ cannot exceed Limits. More info: https://kubernetes.io/docs/concepts/configuration/manage-resources-containers/'
type: object
services:
description: services records the last services of the component.
diff --git a/config/crd/bases/apps.kubeblocks.io_servicedescriptors.yaml b/config/crd/bases/apps.kubeblocks.io_servicedescriptors.yaml
index 2f0dcd73083..653b4a30527 100644
--- a/config/crd/bases/apps.kubeblocks.io_servicedescriptors.yaml
+++ b/config/crd/bases/apps.kubeblocks.io_servicedescriptors.yaml
@@ -2,8 +2,7 @@ apiVersion: apiextensions.k8s.io/v1
kind: CustomResourceDefinition
metadata:
annotations:
- controller-gen.kubebuilder.io/version: v0.9.0
- creationTimestamp: null
+ controller-gen.kubebuilder.io/version: v0.12.1
labels:
app.kubernetes.io/name: kubeblocks
name: servicedescriptors.apps.kubeblocks.io
@@ -97,6 +96,7 @@ spec:
required:
- key
type: object
+ x-kubernetes-map-type: atomic
fieldRef:
description: 'Selects a field of the pod: supports metadata.name,
metadata.namespace, `metadata.labels['''']`, `metadata.annotations['''']`,
@@ -114,6 +114,7 @@ spec:
required:
- fieldPath
type: object
+ x-kubernetes-map-type: atomic
resourceFieldRef:
description: 'Selects a resource of the container: only
resources limits and requests (limits.cpu, limits.memory,
@@ -138,6 +139,7 @@ spec:
required:
- resource
type: object
+ x-kubernetes-map-type: atomic
secretKeyRef:
description: Selects a key of a secret in the pod's namespace
properties:
@@ -157,6 +159,7 @@ spec:
required:
- key
type: object
+ x-kubernetes-map-type: atomic
type: object
type: object
username:
@@ -196,6 +199,7 @@ spec:
required:
- key
type: object
+ x-kubernetes-map-type: atomic
fieldRef:
description: 'Selects a field of the pod: supports metadata.name,
metadata.namespace, `metadata.labels['''']`, `metadata.annotations['''']`,
@@ -213,6 +217,7 @@ spec:
required:
- fieldPath
type: object
+ x-kubernetes-map-type: atomic
resourceFieldRef:
description: 'Selects a resource of the container: only
resources limits and requests (limits.cpu, limits.memory,
@@ -237,6 +242,7 @@ spec:
required:
- resource
type: object
+ x-kubernetes-map-type: atomic
secretKeyRef:
description: Selects a key of a secret in the pod's namespace
properties:
@@ -256,6 +262,7 @@ spec:
required:
- key
type: object
+ x-kubernetes-map-type: atomic
type: object
type: object
type: object
@@ -294,6 +301,7 @@ spec:
required:
- key
type: object
+ x-kubernetes-map-type: atomic
fieldRef:
description: 'Selects a field of the pod: supports metadata.name,
metadata.namespace, `metadata.labels['''']`, `metadata.annotations['''']`,
@@ -311,6 +319,7 @@ spec:
required:
- fieldPath
type: object
+ x-kubernetes-map-type: atomic
resourceFieldRef:
description: 'Selects a resource of the container: only resources
limits and requests (limits.cpu, limits.memory, limits.ephemeral-storage,
@@ -335,6 +344,7 @@ spec:
required:
- resource
type: object
+ x-kubernetes-map-type: atomic
secretKeyRef:
description: Selects a key of a secret in the pod's namespace
properties:
@@ -353,6 +363,7 @@ spec:
required:
- key
type: object
+ x-kubernetes-map-type: atomic
type: object
type: object
port:
@@ -390,6 +401,7 @@ spec:
required:
- key
type: object
+ x-kubernetes-map-type: atomic
fieldRef:
description: 'Selects a field of the pod: supports metadata.name,
metadata.namespace, `metadata.labels['''']`, `metadata.annotations['''']`,
@@ -407,6 +419,7 @@ spec:
required:
- fieldPath
type: object
+ x-kubernetes-map-type: atomic
resourceFieldRef:
description: 'Selects a resource of the container: only resources
limits and requests (limits.cpu, limits.memory, limits.ephemeral-storage,
@@ -431,6 +444,7 @@ spec:
required:
- resource
type: object
+ x-kubernetes-map-type: atomic
secretKeyRef:
description: Selects a key of a secret in the pod's namespace
properties:
@@ -449,6 +463,7 @@ spec:
required:
- key
type: object
+ x-kubernetes-map-type: atomic
type: object
type: object
serviceKind:
diff --git a/config/crd/bases/dataprotection.kubeblocks.io_actionsets.yaml b/config/crd/bases/dataprotection.kubeblocks.io_actionsets.yaml
new file mode 100644
index 00000000000..85c9df43f48
--- /dev/null
+++ b/config/crd/bases/dataprotection.kubeblocks.io_actionsets.yaml
@@ -0,0 +1,554 @@
+apiVersion: apiextensions.k8s.io/v1
+kind: CustomResourceDefinition
+metadata:
+ annotations:
+ controller-gen.kubebuilder.io/version: v0.12.1
+ labels:
+ app.kubernetes.io/name: kubeblocks
+ name: actionsets.dataprotection.kubeblocks.io
+spec:
+ group: dataprotection.kubeblocks.io
+ names:
+ categories:
+ - kubeblocks
+ kind: ActionSet
+ listKind: ActionSetList
+ plural: actionsets
+ shortNames:
+ - as
+ singular: actionset
+ scope: Cluster
+ versions:
+ - additionalPrinterColumns:
+ - jsonPath: .spec.backupType
+ name: BACKUP-TYPE
+ type: string
+ - jsonPath: .status.phase
+ name: STATUS
+ type: string
+ - jsonPath: .metadata.creationTimestamp
+ name: AGE
+ type: date
+ name: v1alpha1
+ schema:
+ openAPIV3Schema:
+ description: ActionSet is the Schema for the actionsets API
+ properties:
+ apiVersion:
+ description: 'APIVersion defines the versioned schema of this representation
+ of an object. Servers should convert recognized schemas to the latest
+ internal value, and may reject unrecognized values. More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#resources'
+ type: string
+ kind:
+ description: 'Kind is a string value representing the REST resource this
+ object represents. Servers may infer this from the endpoint the client
+ submits requests to. Cannot be updated. In CamelCase. More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#types-kinds'
+ type: string
+ metadata:
+ type: object
+ spec:
+ description: ActionSetSpec defines the desired state of ActionSet
+ properties:
+ backup:
+ description: backup specifies the backup action.
+ properties:
+ backupData:
+ description: backupData specifies the backup data action.
+ properties:
+ command:
+ description: command specifies the commands to back up the
+ volume data.
+ items:
+ type: string
+ type: array
+ image:
+ description: image specifies the image of backup container.
+ type: string
+ onError:
+ default: Fail
+ description: OnError specifies how should behave if it encounters
+ an error executing this action.
+ enum:
+ - Continue
+ - Fail
+ type: string
+ runOnTargetPodNode:
+ default: false
+ description: runOnTargetPodNode specifies whether to run the
+ job workload on the target pod node. If backup container
+ should mount the target pod's volume, this field should
+ be set to true.
+ type: boolean
+ syncProgress:
+ description: syncProgress specifies whether to sync the backup
+ progress and its interval seconds.
+ properties:
+ enabled:
+ description: enabled specifies whether to sync the backup
+ progress. If enabled, a sidecar container will be created
+ to sync the backup progress to the Backup CR status.
+ type: boolean
+ intervalSeconds:
+ default: 60
+ description: intervalSeconds specifies the interval seconds
+ to sync the backup progress.
+ format: int32
+ type: integer
+ type: object
+ required:
+ - command
+ - image
+ type: object
+ postBackup:
+ description: postBackup specifies a hook that should be executed
+ after the backup.
+ items:
+ description: ActionSpec defines an action that should be executed.
+ Only one of the fields may be set.
+ properties:
+ exec:
+ description: exec specifies the action should be executed
+ by the pod exec API in a container.
+ properties:
+ command:
+ description: Command is the command and arguments to
+ execute.
+ items:
+ type: string
+ minItems: 1
+ type: array
+ container:
+ description: container is the container in the pod where
+ the command should be executed. If not specified,
+ the pod's first container is used.
+ type: string
+ onError:
+ default: Fail
+ description: OnError specifies how should behave if
+ it encounters an error executing this action.
+ enum:
+ - Continue
+ - Fail
+ type: string
+ timeout:
+ description: Timeout defines the maximum amount of time
+ should wait for the hook to complete before considering
+ the execution a failure.
+ type: string
+ required:
+ - command
+ type: object
+ job:
+ description: job specifies the action should be executed
+ by a Kubernetes Job.
+ properties:
+ command:
+ description: command specifies the commands to back
+ up the volume data.
+ items:
+ type: string
+ type: array
+ image:
+ description: image specifies the image of backup container.
+ type: string
+ onError:
+ default: Fail
+ description: OnError specifies how should behave if
+ it encounters an error executing this action.
+ enum:
+ - Continue
+ - Fail
+ type: string
+ runOnTargetPodNode:
+ default: false
+ description: runOnTargetPodNode specifies whether to
+ run the job workload on the target pod node. If backup
+ container should mount the target pod's volume, this
+ field should be set to true.
+ type: boolean
+ required:
+ - command
+ - image
+ type: object
+ type: object
+ type: array
+ preBackup:
+ description: preBackup specifies a hook that should be executed
+ before the backup.
+ items:
+ description: ActionSpec defines an action that should be executed.
+ Only one of the fields may be set.
+ properties:
+ exec:
+ description: exec specifies the action should be executed
+ by the pod exec API in a container.
+ properties:
+ command:
+ description: Command is the command and arguments to
+ execute.
+ items:
+ type: string
+ minItems: 1
+ type: array
+ container:
+ description: container is the container in the pod where
+ the command should be executed. If not specified,
+ the pod's first container is used.
+ type: string
+ onError:
+ default: Fail
+ description: OnError specifies how should behave if
+ it encounters an error executing this action.
+ enum:
+ - Continue
+ - Fail
+ type: string
+ timeout:
+ description: Timeout defines the maximum amount of time
+ should wait for the hook to complete before considering
+ the execution a failure.
+ type: string
+ required:
+ - command
+ type: object
+ job:
+ description: job specifies the action should be executed
+ by a Kubernetes Job.
+ properties:
+ command:
+ description: command specifies the commands to back
+ up the volume data.
+ items:
+ type: string
+ type: array
+ image:
+ description: image specifies the image of backup container.
+ type: string
+ onError:
+ default: Fail
+ description: OnError specifies how should behave if
+ it encounters an error executing this action.
+ enum:
+ - Continue
+ - Fail
+ type: string
+ runOnTargetPodNode:
+ default: false
+ description: runOnTargetPodNode specifies whether to
+ run the job workload on the target pod node. If backup
+ container should mount the target pod's volume, this
+ field should be set to true.
+ type: boolean
+ required:
+ - command
+ - image
+ type: object
+ type: object
+ type: array
+ type: object
+ backupType:
+ allOf:
+ - enum:
+ - Full
+ - Incremental
+ - Differential
+ - Continuous
+ - enum:
+ - Full
+ - Incremental
+ - Differential
+ - Continuous
+ default: Full
+ description: 'backupType specifies the backup type, supported values:
+ Full, Continuous. Full means full backup. Incremental means back
+ up data that have changed since the last backup (full or incremental).
+ Differential means back up data that have changed since the last
+ full backup. Continuous will back up the transaction log continuously,
+ the PITR (Point in Time Recovery). can be performed based on the
+ continuous backup and full backup.'
+ type: string
+ env:
+ description: List of environment variables to set in the container.
+ items:
+ description: EnvVar represents an environment variable present in
+ a Container.
+ properties:
+ name:
+ description: Name of the environment variable. Must be a C_IDENTIFIER.
+ type: string
+ value:
+ description: 'Variable references $(VAR_NAME) are expanded using
+ the previously defined environment variables in the container
+ and any service environment variables. If a variable cannot
+ be resolved, the reference in the input string will be unchanged.
+ Double $$ are reduced to a single $, which allows for escaping
+ the $(VAR_NAME) syntax: i.e. "$$(VAR_NAME)" will produce the
+ string literal "$(VAR_NAME)". Escaped references will never
+ be expanded, regardless of whether the variable exists or
+ not. Defaults to "".'
+ type: string
+ valueFrom:
+ description: Source for the environment variable's value. Cannot
+ be used if value is not empty.
+ properties:
+ configMapKeyRef:
+ description: Selects a key of a ConfigMap.
+ properties:
+ key:
+ description: The key to select.
+ type: string
+ name:
+ description: 'Name of the referent. More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names
+ TODO: Add other useful fields. apiVersion, kind, uid?'
+ type: string
+ optional:
+ description: Specify whether the ConfigMap or its key
+ must be defined
+ type: boolean
+ required:
+ - key
+ type: object
+ x-kubernetes-map-type: atomic
+ fieldRef:
+ description: 'Selects a field of the pod: supports metadata.name,
+ metadata.namespace, `metadata.labels['''']`, `metadata.annotations['''']`,
+ spec.nodeName, spec.serviceAccountName, status.hostIP,
+ status.podIP, status.podIPs.'
+ properties:
+ apiVersion:
+ description: Version of the schema the FieldPath is
+ written in terms of, defaults to "v1".
+ type: string
+ fieldPath:
+ description: Path of the field to select in the specified
+ API version.
+ type: string
+ required:
+ - fieldPath
+ type: object
+ x-kubernetes-map-type: atomic
+ resourceFieldRef:
+ description: 'Selects a resource of the container: only
+ resources limits and requests (limits.cpu, limits.memory,
+ limits.ephemeral-storage, requests.cpu, requests.memory
+ and requests.ephemeral-storage) are currently supported.'
+ properties:
+ containerName:
+ description: 'Container name: required for volumes,
+ optional for env vars'
+ type: string
+ divisor:
+ anyOf:
+ - type: integer
+ - type: string
+ description: Specifies the output format of the exposed
+ resources, defaults to "1"
+ pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$
+ x-kubernetes-int-or-string: true
+ resource:
+ description: 'Required: resource to select'
+ type: string
+ required:
+ - resource
+ type: object
+ x-kubernetes-map-type: atomic
+ secretKeyRef:
+ description: Selects a key of a secret in the pod's namespace
+ properties:
+ key:
+ description: The key of the secret to select from. Must
+ be a valid secret key.
+ type: string
+ name:
+ description: 'Name of the referent. More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names
+ TODO: Add other useful fields. apiVersion, kind, uid?'
+ type: string
+ optional:
+ description: Specify whether the Secret or its key must
+ be defined
+ type: boolean
+ required:
+ - key
+ type: object
+ x-kubernetes-map-type: atomic
+ type: object
+ required:
+ - name
+ type: object
+ type: array
+ x-kubernetes-preserve-unknown-fields: true
+ envFrom:
+ description: List of sources to populate environment variables in
+ the container. The keys defined within a source must be a C_IDENTIFIER.
+ All invalid keys will be reported as an event when the container
+ is starting. When a key exists in multiple sources, the value associated
+ with the last source will take precedence. Values defined by an
+ Env with a duplicate key will take precedence. Cannot be updated.
+ items:
+ description: EnvFromSource represents the source of a set of ConfigMaps
+ properties:
+ configMapRef:
+ description: The ConfigMap to select from
+ properties:
+ name:
+ description: 'Name of the referent. More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names
+ TODO: Add other useful fields. apiVersion, kind, uid?'
+ type: string
+ optional:
+ description: Specify whether the ConfigMap must be defined
+ type: boolean
+ type: object
+ x-kubernetes-map-type: atomic
+ prefix:
+ description: An optional identifier to prepend to each key in
+ the ConfigMap. Must be a C_IDENTIFIER.
+ type: string
+ secretRef:
+ description: The Secret to select from
+ properties:
+ name:
+ description: 'Name of the referent. More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names
+ TODO: Add other useful fields. apiVersion, kind, uid?'
+ type: string
+ optional:
+ description: Specify whether the Secret must be defined
+ type: boolean
+ type: object
+ x-kubernetes-map-type: atomic
+ type: object
+ type: array
+ x-kubernetes-preserve-unknown-fields: true
+ restore:
+ description: restore specifies the restore action.
+ properties:
+ postReady:
+ description: postReady specifies the action to execute after the
+ data is ready.
+ items:
+ description: ActionSpec defines an action that should be executed.
+ Only one of the fields may be set.
+ properties:
+ exec:
+ description: exec specifies the action should be executed
+ by the pod exec API in a container.
+ properties:
+ command:
+ description: Command is the command and arguments to
+ execute.
+ items:
+ type: string
+ minItems: 1
+ type: array
+ container:
+ description: container is the container in the pod where
+ the command should be executed. If not specified,
+ the pod's first container is used.
+ type: string
+ onError:
+ default: Fail
+ description: OnError specifies how should behave if
+ it encounters an error executing this action.
+ enum:
+ - Continue
+ - Fail
+ type: string
+ timeout:
+ description: Timeout defines the maximum amount of time
+ should wait for the hook to complete before considering
+ the execution a failure.
+ type: string
+ required:
+ - command
+ type: object
+ job:
+ description: job specifies the action should be executed
+ by a Kubernetes Job.
+ properties:
+ command:
+ description: command specifies the commands to back
+ up the volume data.
+ items:
+ type: string
+ type: array
+ image:
+ description: image specifies the image of backup container.
+ type: string
+ onError:
+ default: Fail
+ description: OnError specifies how should behave if
+ it encounters an error executing this action.
+ enum:
+ - Continue
+ - Fail
+ type: string
+ runOnTargetPodNode:
+ default: false
+ description: runOnTargetPodNode specifies whether to
+ run the job workload on the target pod node. If backup
+ container should mount the target pod's volume, this
+ field should be set to true.
+ type: boolean
+ required:
+ - command
+ - image
+ type: object
+ type: object
+ type: array
+ prepareData:
+ description: prepareData specifies the action to prepare data.
+ properties:
+ command:
+ description: command specifies the commands to back up the
+ volume data.
+ items:
+ type: string
+ type: array
+ image:
+ description: image specifies the image of backup container.
+ type: string
+ onError:
+ default: Fail
+ description: OnError specifies how should behave if it encounters
+ an error executing this action.
+ enum:
+ - Continue
+ - Fail
+ type: string
+ runOnTargetPodNode:
+ default: false
+ description: runOnTargetPodNode specifies whether to run the
+ job workload on the target pod node. If backup container
+ should mount the target pod's volume, this field should
+ be set to true.
+ type: boolean
+ required:
+ - command
+ - image
+ type: object
+ type: object
+ required:
+ - backupType
+ type: object
+ status:
+ description: ActionSetStatus defines the observed state of ActionSet
+ properties:
+ message:
+ description: A human-readable message indicating details about why
+ the ActionSet is in this phase.
+ type: string
+ observedGeneration:
+ description: generation number
+ format: int64
+ type: integer
+ phase:
+ description: phase - in list of [Available,Unavailable]
+ enum:
+ - Available
+ - Unavailable
+ type: string
+ type: object
+ type: object
+ served: true
+ storage: true
+ subresources:
+ status: {}
diff --git a/config/crd/bases/dataprotection.kubeblocks.io_backuppolicies.yaml b/config/crd/bases/dataprotection.kubeblocks.io_backuppolicies.yaml
index f3261eca785..ec61183f863 100644
--- a/config/crd/bases/dataprotection.kubeblocks.io_backuppolicies.yaml
+++ b/config/crd/bases/dataprotection.kubeblocks.io_backuppolicies.yaml
@@ -2,8 +2,7 @@ apiVersion: apiextensions.k8s.io/v1
kind: CustomResourceDefinition
metadata:
annotations:
- controller-gen.kubebuilder.io/version: v0.9.0
- creationTimestamp: null
+ controller-gen.kubebuilder.io/version: v0.12.1
labels:
app.kubernetes.io/name: kubeblocks
name: backuppolicies.dataprotection.kubeblocks.io
@@ -21,20 +20,19 @@ spec:
scope: Namespaced
versions:
- additionalPrinterColumns:
+ - jsonPath: .spec.backupRepoName
+ name: BACKUP-REPO
+ type: string
- jsonPath: .status.phase
name: STATUS
type: string
- - jsonPath: .status.lastScheduleTime
- name: LAST SCHEDULE
- type: string
- jsonPath: .metadata.creationTimestamp
name: AGE
type: date
name: v1alpha1
schema:
openAPIV3Schema:
- description: BackupPolicy is the Schema for the backuppolicies API (defined
- by User)
+ description: BackupPolicy is the Schema for the backuppolicies API.
properties:
apiVersion:
description: 'APIVersion defines the versioned schema of this representation
@@ -51,540 +49,389 @@ spec:
spec:
description: BackupPolicySpec defines the desired state of BackupPolicy
properties:
- datafile:
- description: the policy for datafile backup.
- properties:
- backupRepoName:
- description: refer to BackupRepo and the backup data will be stored
- in the corresponding repo.
- pattern: ^[a-z0-9]([a-z0-9\.\-]*[a-z0-9])?$
- type: string
- backupStatusUpdates:
- description: define how to update metadata for backup status.
- items:
- properties:
- containerName:
- description: which container name that kubectl can execute.
- type: string
- path:
- description: 'specify the json path of backup object for
- patch. example: manifests.backupLog -- means patch the
- backup json path of status.manifests.backupLog.'
- type: string
- script:
- description: the shell Script commands to collect backup
- status metadata. The script must exist in the container
- of ContainerName and the output format must be set to
- JSON. Note that outputting to stderr may cause the result
- format to not be in JSON.
- type: string
- updateStage:
- description: 'when to update the backup status, pre: before
- backup, post: after backup'
- enum:
- - pre
- - post
- type: string
- useTargetPodServiceAccount:
- description: useTargetPodServiceAccount defines whether
- this job requires the service account of the backup target
- pod. if true, will use the service account of the backup
- target pod. otherwise, will use the system service account.
- type: boolean
- required:
- - updateStage
- type: object
- type: array
- backupToolName:
- description: which backup tool to perform database backup, only
- support one tool.
- pattern: ^[a-z0-9]([a-z0-9\.\-]*[a-z0-9])?$
- type: string
- backupsHistoryLimit:
- default: 7
- description: the number of automatic backups to retain. Value
- must be non-negative integer. 0 means NO limit on the number
- of backups.
- format: int32
- type: integer
- onFailAttempted:
- description: count of backup stop retries on fail.
- format: int32
- type: integer
- persistentVolumeClaim:
- description: refer to PersistentVolumeClaim and the backup data
- will be stored in the corresponding persistent volume.
- properties:
- createPolicy:
- default: IfNotPresent
- description: 'createPolicy defines the policy for creating
- the PersistentVolumeClaim, enum values: - Never: do nothing
- if the PersistentVolumeClaim not exists. - IfNotPresent:
- create the PersistentVolumeClaim if not present and the
- accessModes only contains ''ReadWriteMany''.'
- enum:
- - IfNotPresent
- - Never
- type: string
- initCapacity:
- anyOf:
- - type: integer
- - type: string
- description: initCapacity represents the init storage size
- of the PersistentVolumeClaim which should be created if
- not exist. and the default value is 100Gi if it is empty.
- pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$
- x-kubernetes-int-or-string: true
- name:
- description: the name of PersistentVolumeClaim to store backup
- data.
- type: string
- persistentVolumeConfigMap:
- description: 'persistentVolumeConfigMap references the configmap
- which contains a persistentVolume template. key must be
- "persistentVolume" and value is the "PersistentVolume" struct.
- support the following built-in Objects: - $(GENERATE_NAME):
- generate a specific format "`PVC NAME`-`PVC NAMESPACE`".
- if the PersistentVolumeClaim not exists and CreatePolicy
- is "IfNotPresent", the controller will create it by this
- template. this is a mutually exclusive setting with "storageClassName".'
+ backoffLimit:
+ description: Specifies the number of retries before marking the backup
+ failed.
+ format: int32
+ maximum: 10
+ minimum: 0
+ type: integer
+ backupMethods:
+ description: backupMethods defines the backup methods.
+ items:
+ description: BackupMethod defines the backup method.
+ properties:
+ actionSetName:
+ description: actionSetName refers to the ActionSet object that
+ defines the backup actions. For volume snapshot backup, the
+ actionSet is not required, the controller will use the CSI
+ volume snapshotter to create the snapshot.
+ type: string
+ env:
+ description: env specifies the environment variables for the
+ backup workload.
+ items:
+ description: EnvVar represents an environment variable present
+ in a Container.
properties:
name:
- description: the name of the persistentVolume ConfigMap.
- pattern: ^[a-z0-9]([a-z0-9\.\-]*[a-z0-9])?$
+ description: Name of the environment variable. Must be
+ a C_IDENTIFIER.
type: string
- namespace:
- description: the namespace of the persistentVolume ConfigMap.
- pattern: ^[a-z0-9]([a-z0-9\-]*[a-z0-9])?$
+ value:
+ description: 'Variable references $(VAR_NAME) are expanded
+ using the previously defined environment variables in
+ the container and any service environment variables.
+ If a variable cannot be resolved, the reference in the
+ input string will be unchanged. Double $$ are reduced
+ to a single $, which allows for escaping the $(VAR_NAME)
+ syntax: i.e. "$$(VAR_NAME)" will produce the string
+ literal "$(VAR_NAME)". Escaped references will never
+ be expanded, regardless of whether the variable exists
+ or not. Defaults to "".'
type: string
- required:
- - name
- - namespace
- type: object
- storageClassName:
- description: storageClassName is the name of the StorageClass
- required by the claim.
- pattern: ^[a-z0-9]([a-z0-9\.\-]*[a-z0-9])?$
- type: string
- type: object
- target:
- description: target database cluster for backup.
- properties:
- labelsSelector:
- description: labelsSelector is used to find matching pods.
- Pods that match this label selector are counted to determine
- the number of pods in their corresponding topology domain.
- properties:
- matchExpressions:
- description: matchExpressions is a list of label selector
- requirements. The requirements are ANDed.
- items:
- description: A label selector requirement is a selector
- that contains values, a key, and an operator that
- relates the key and values.
- properties:
- key:
- description: key is the label key that the selector
- applies to.
- type: string
- operator:
- description: operator represents a key's relationship
- to a set of values. Valid operators are In, NotIn,
- Exists and DoesNotExist.
- type: string
- values:
- description: values is an array of string values.
- If the operator is In or NotIn, the values array
- must be non-empty. If the operator is Exists or
- DoesNotExist, the values array must be empty.
- This array is replaced during a strategic merge
- patch.
- items:
+ valueFrom:
+ description: Source for the environment variable's value.
+ Cannot be used if value is not empty.
+ properties:
+ configMapKeyRef:
+ description: Selects a key of a ConfigMap.
+ properties:
+ key:
+ description: The key to select.
type: string
- type: array
- required:
- - key
- - operator
- type: object
- type: array
- matchLabels:
- additionalProperties:
- type: string
- description: matchLabels is a map of {key,value} pairs.
- A single {key,value} in the matchLabels map is equivalent
- to an element of matchExpressions, whose key field is
- "key", the operator is "In", and the values array contains
- only "value". The requirements are ANDed.
+ name:
+ description: 'Name of the referent. More info:
+ https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names
+ TODO: Add other useful fields. apiVersion, kind,
+ uid?'
+ type: string
+ optional:
+ description: Specify whether the ConfigMap or
+ its key must be defined
+ type: boolean
+ required:
+ - key
+ type: object
+ x-kubernetes-map-type: atomic
+ fieldRef:
+ description: 'Selects a field of the pod: supports
+ metadata.name, metadata.namespace, `metadata.labels['''']`,
+ `metadata.annotations['''']`, spec.nodeName,
+ spec.serviceAccountName, status.hostIP, status.podIP,
+ status.podIPs.'
+ properties:
+ apiVersion:
+ description: Version of the schema the FieldPath
+ is written in terms of, defaults to "v1".
+ type: string
+ fieldPath:
+ description: Path of the field to select in the
+ specified API version.
+ type: string
+ required:
+ - fieldPath
+ type: object
+ x-kubernetes-map-type: atomic
+ resourceFieldRef:
+ description: 'Selects a resource of the container:
+ only resources limits and requests (limits.cpu,
+ limits.memory, limits.ephemeral-storage, requests.cpu,
+ requests.memory and requests.ephemeral-storage)
+ are currently supported.'
+ properties:
+ containerName:
+ description: 'Container name: required for volumes,
+ optional for env vars'
+ type: string
+ divisor:
+ anyOf:
+ - type: integer
+ - type: string
+ description: Specifies the output format of the
+ exposed resources, defaults to "1"
+ pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$
+ x-kubernetes-int-or-string: true
+ resource:
+ description: 'Required: resource to select'
+ type: string
+ required:
+ - resource
+ type: object
+ x-kubernetes-map-type: atomic
+ secretKeyRef:
+ description: Selects a key of a secret in the pod's
+ namespace
+ properties:
+ key:
+ description: The key of the secret to select from. Must
+ be a valid secret key.
+ type: string
+ name:
+ description: 'Name of the referent. More info:
+ https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names
+ TODO: Add other useful fields. apiVersion, kind,
+ uid?'
+ type: string
+ optional:
+ description: Specify whether the Secret or its
+ key must be defined
+ type: boolean
+ required:
+ - key
+ type: object
+ x-kubernetes-map-type: atomic
type: object
- type: object
- x-kubernetes-preserve-unknown-fields: true
- secret:
- description: secret is used to connect to the target database
- cluster. If not set, secret will be inherited from backup
- policy template. if still not set, the controller will check
- if any system account for dataprotection has been created.
- properties:
- name:
- description: the secret name
- pattern: ^[a-z0-9]([a-z0-9\.\-]*[a-z0-9])?$
- type: string
- passwordKey:
- default: password
- description: passwordKey the map key of the password in
- the connection credential secret
- type: string
- usernameKey:
- default: username
- description: usernameKey the map key of the user in the
- connection credential secret
- type: string
required:
- name
type: object
- required:
- - labelsSelector
- type: object
- required:
- - target
- type: object
- logfile:
- description: the policy for logfile backup.
- properties:
- backupRepoName:
- description: refer to BackupRepo and the backup data will be stored
- in the corresponding repo.
- pattern: ^[a-z0-9]([a-z0-9\.\-]*[a-z0-9])?$
- type: string
- backupStatusUpdates:
- description: define how to update metadata for backup status.
- items:
+ type: array
+ name:
+ description: the name of backup method.
+ pattern: ^[a-z0-9]([a-z0-9\.\-]*[a-z0-9])?$
+ type: string
+ runtimeSettings:
+ description: runtimeSettings specifies runtime settings for
+ the backup workload container.
properties:
- containerName:
- description: which container name that kubectl can execute.
- type: string
- path:
- description: 'specify the json path of backup object for
- patch. example: manifests.backupLog -- means patch the
- backup json path of status.manifests.backupLog.'
- type: string
- script:
- description: the shell Script commands to collect backup
- status metadata. The script must exist in the container
- of ContainerName and the output format must be set to
- JSON. Note that outputting to stderr may cause the result
- format to not be in JSON.
- type: string
- updateStage:
- description: 'when to update the backup status, pre: before
- backup, post: after backup'
- enum:
- - pre
- - post
- type: string
- useTargetPodServiceAccount:
- description: useTargetPodServiceAccount defines whether
- this job requires the service account of the backup target
- pod. if true, will use the service account of the backup
- target pod. otherwise, will use the system service account.
- type: boolean
- required:
- - updateStage
- type: object
- type: array
- backupToolName:
- description: which backup tool to perform database backup, only
- support one tool.
- pattern: ^[a-z0-9]([a-z0-9\.\-]*[a-z0-9])?$
- type: string
- backupsHistoryLimit:
- default: 7
- description: the number of automatic backups to retain. Value
- must be non-negative integer. 0 means NO limit on the number
- of backups.
- format: int32
- type: integer
- onFailAttempted:
- description: count of backup stop retries on fail.
- format: int32
- type: integer
- persistentVolumeClaim:
- description: refer to PersistentVolumeClaim and the backup data
- will be stored in the corresponding persistent volume.
- properties:
- createPolicy:
- default: IfNotPresent
- description: 'createPolicy defines the policy for creating
- the PersistentVolumeClaim, enum values: - Never: do nothing
- if the PersistentVolumeClaim not exists. - IfNotPresent:
- create the PersistentVolumeClaim if not present and the
- accessModes only contains ''ReadWriteMany''.'
- enum:
- - IfNotPresent
- - Never
- type: string
- initCapacity:
- anyOf:
- - type: integer
- - type: string
- description: initCapacity represents the init storage size
- of the PersistentVolumeClaim which should be created if
- not exist. and the default value is 100Gi if it is empty.
- pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$
- x-kubernetes-int-or-string: true
- name:
- description: the name of PersistentVolumeClaim to store backup
- data.
- type: string
- persistentVolumeConfigMap:
- description: 'persistentVolumeConfigMap references the configmap
- which contains a persistentVolume template. key must be
- "persistentVolume" and value is the "PersistentVolume" struct.
- support the following built-in Objects: - $(GENERATE_NAME):
- generate a specific format "`PVC NAME`-`PVC NAMESPACE`".
- if the PersistentVolumeClaim not exists and CreatePolicy
- is "IfNotPresent", the controller will create it by this
- template. this is a mutually exclusive setting with "storageClassName".'
- properties:
- name:
- description: the name of the persistentVolume ConfigMap.
- pattern: ^[a-z0-9]([a-z0-9\.\-]*[a-z0-9])?$
- type: string
- namespace:
- description: the namespace of the persistentVolume ConfigMap.
- pattern: ^[a-z0-9]([a-z0-9\-]*[a-z0-9])?$
- type: string
- required:
- - name
- - namespace
- type: object
- storageClassName:
- description: storageClassName is the name of the StorageClass
- required by the claim.
- pattern: ^[a-z0-9]([a-z0-9\.\-]*[a-z0-9])?$
- type: string
- type: object
- target:
- description: target database cluster for backup.
- properties:
- labelsSelector:
- description: labelsSelector is used to find matching pods.
- Pods that match this label selector are counted to determine
- the number of pods in their corresponding topology domain.
- properties:
- matchExpressions:
- description: matchExpressions is a list of label selector
- requirements. The requirements are ANDed.
- items:
- description: A label selector requirement is a selector
- that contains values, a key, and an operator that
- relates the key and values.
- properties:
- key:
- description: key is the label key that the selector
- applies to.
- type: string
- operator:
- description: operator represents a key's relationship
- to a set of values. Valid operators are In, NotIn,
- Exists and DoesNotExist.
- type: string
- values:
- description: values is an array of string values.
- If the operator is In or NotIn, the values array
- must be non-empty. If the operator is Exists or
- DoesNotExist, the values array must be empty.
- This array is replaced during a strategic merge
- patch.
- items:
+ resources:
+ description: 'resources specifies the resource required
+ by container. More info: https://kubernetes.io/docs/concepts/configuration/manage-compute-resources-container/'
+ properties:
+ claims:
+ description: "Claims lists the names of resources, defined
+ in spec.resourceClaims, that are used by this container.
+ \n This is an alpha field and requires enabling the
+ DynamicResourceAllocation feature gate. \n This field
+ is immutable. It can only be set for containers."
+ items:
+ description: ResourceClaim references one entry in
+ PodSpec.ResourceClaims.
+ properties:
+ name:
+ description: Name must match the name of one entry
+ in pod.spec.resourceClaims of the Pod where
+ this field is used. It makes that resource available
+ inside a container.
type: string
- type: array
- required:
- - key
- - operator
+ required:
+ - name
+ type: object
+ type: array
+ x-kubernetes-list-map-keys:
+ - name
+ x-kubernetes-list-type: map
+ limits:
+ additionalProperties:
+ anyOf:
+ - type: integer
+ - type: string
+ pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$
+ x-kubernetes-int-or-string: true
+ description: 'Limits describes the maximum amount of
+ compute resources allowed. More info: https://kubernetes.io/docs/concepts/configuration/manage-resources-containers/'
type: object
- type: array
- matchLabels:
- additionalProperties:
- type: string
- description: matchLabels is a map of {key,value} pairs.
- A single {key,value} in the matchLabels map is equivalent
- to an element of matchExpressions, whose key field is
- "key", the operator is "In", and the values array contains
- only "value". The requirements are ANDed.
+ requests:
+ additionalProperties:
+ anyOf:
+ - type: integer
+ - type: string
+ pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$
+ x-kubernetes-int-or-string: true
+ description: 'Requests describes the minimum amount
+ of compute resources required. If Requests is omitted
+ for a container, it defaults to Limits if that is
+ explicitly specified, otherwise to an implementation-defined
+ value. Requests cannot exceed Limits. More info: https://kubernetes.io/docs/concepts/configuration/manage-resources-containers/'
+ type: object
+ type: object
+ type: object
+ snapshotVolumes:
+ default: false
+ description: snapshotVolumes specifies whether to take snapshots
+ of persistent volumes. if true, the BackupScript is not required,
+ the controller will use the CSI volume snapshotter to create
+ the snapshot.
+ type: boolean
+ targetVolumes:
+ description: targetVolumes specifies which volumes from the
+ target should be mounted in the backup workload.
+ properties:
+ volumeMounts:
+ description: volumeMounts specifies the mount for the volumes
+ specified in `Volumes` section.
+ items:
+ description: VolumeMount describes a mounting of a Volume
+ within a container.
+ properties:
+ mountPath:
+ description: Path within the container at which the
+ volume should be mounted. Must not contain ':'.
+ type: string
+ mountPropagation:
+ description: mountPropagation determines how mounts
+ are propagated from the host to container and the
+ other way around. When not set, MountPropagationNone
+ is used. This field is beta in 1.10.
+ type: string
+ name:
+ description: This must match the Name of a Volume.
+ type: string
+ readOnly:
+ description: Mounted read-only if true, read-write
+ otherwise (false or unspecified). Defaults to false.
+ type: boolean
+ subPath:
+ description: Path within the volume from which the
+ container's volume should be mounted. Defaults to
+ "" (volume's root).
+ type: string
+ subPathExpr:
+ description: Expanded path within the volume from
+ which the container's volume should be mounted.
+ Behaves similarly to SubPath but environment variable
+ references $(VAR_NAME) are expanded using the container's
+ environment. Defaults to "" (volume's root). SubPathExpr
+ and SubPath are mutually exclusive.
+ type: string
+ required:
+ - mountPath
+ - name
type: object
- type: object
- x-kubernetes-preserve-unknown-fields: true
- secret:
- description: secret is used to connect to the target database
- cluster. If not set, secret will be inherited from backup
- policy template. if still not set, the controller will check
- if any system account for dataprotection has been created.
- properties:
- name:
- description: the secret name
- pattern: ^[a-z0-9]([a-z0-9\.\-]*[a-z0-9])?$
+ type: array
+ volumes:
+ description: Volumes indicates the list of volumes of targeted
+ application that should be mounted on the backup job.
+ items:
type: string
- passwordKey:
- default: password
- description: passwordKey the map key of the password in
- the connection credential secret
- type: string
- usernameKey:
- default: username
- description: usernameKey the map key of the user in the
- connection credential secret
- type: string
- required:
- - name
- type: object
- required:
- - labelsSelector
- type: object
- required:
- - target
- type: object
- retention:
- description: retention describe how long the Backup should be retained.
- if not set, will be retained forever.
- properties:
- ttl:
- description: ttl is a time string ending with the 'd'|'D'|'h'|'H'
- character to describe how long the Backup should be retained.
- if not set, will be retained forever.
- pattern: ^\d+[d|D|h|H]$
- type: string
- type: object
- schedule:
- description: schedule policy for backup.
+ type: array
+ type: object
+ required:
+ - name
+ type: object
+ type: array
+ backupRepoName:
+ description: backupRepoName is the name of BackupRepo and the backup
+ data will be stored in this repository. If not set, will be stored
+ in the default backup repository.
+ pattern: ^[a-z0-9]([a-z0-9\.\-]*[a-z0-9])?$
+ type: string
+ pathPrefix:
+ description: pathPrefix is the directory inside the backup repository
+ to store the backup content. It is a relative to the path of the
+ backup repository.
+ type: string
+ target:
+ description: target specifies the target information to back up.
properties:
- datafile:
- description: schedule policy for datafile backup.
+ connectionCredential:
+ description: connectionCredential specifies the connection credential
+ to connect to the target database cluster.
properties:
- cronExpression:
- description: the cron expression for schedule, the timezone
- is in UTC. see https://en.wikipedia.org/wiki/Cron.
+ hostKey:
+ description: hostKey specifies the map key of the host in
+ the connection credential secret.
type: string
- enable:
- description: enable or disable the schedule.
- type: boolean
- required:
- - cronExpression
- - enable
- type: object
- logfile:
- description: schedule policy for logfile backup.
- properties:
- cronExpression:
- description: the cron expression for schedule, the timezone
- is in UTC. see https://en.wikipedia.org/wiki/Cron.
+ passwordKey:
+ default: password
+ description: passwordKey specifies the map key of the password
+ in the connection credential secret.
+ type: string
+ portKey:
+ default: port
+ description: portKey specifies the map key of the port in
+ the connection credential secret.
+ type: string
+ secretName:
+ description: secretName refers to the Secret object that contains
+ the connection credential.
+ type: string
+ usernameKey:
+ default: username
+ description: usernameKey specifies the map key of the user
+ in the connection credential secret.
type: string
- enable:
- description: enable or disable the schedule.
- type: boolean
- required:
- - cronExpression
- - enable
type: object
- snapshot:
- description: schedule policy for snapshot backup.
+ podSelector:
+ description: podSelector is used to find the target pod. The volumes
+ of the target pod will be backed up.
properties:
- cronExpression:
- description: the cron expression for schedule, the timezone
- is in UTC. see https://en.wikipedia.org/wiki/Cron.
+ matchExpressions:
+ description: matchExpressions is a list of label selector
+ requirements. The requirements are ANDed.
+ items:
+ description: A label selector requirement is a selector
+ that contains values, a key, and an operator that relates
+ the key and values.
+ properties:
+ key:
+ description: key is the label key that the selector
+ applies to.
+ type: string
+ operator:
+ description: operator represents a key's relationship
+ to a set of values. Valid operators are In, NotIn,
+ Exists and DoesNotExist.
+ type: string
+ values:
+ description: values is an array of string values. If
+ the operator is In or NotIn, the values array must
+ be non-empty. If the operator is Exists or DoesNotExist,
+ the values array must be empty. This array is replaced
+ during a strategic merge patch.
+ items:
+ type: string
+ type: array
+ required:
+ - key
+ - operator
+ type: object
+ type: array
+ matchLabels:
+ additionalProperties:
+ type: string
+ description: matchLabels is a map of {key,value} pairs. A
+ single {key,value} in the matchLabels map is equivalent
+ to an element of matchExpressions, whose key field is "key",
+ the operator is "In", and the values array contains only
+ "value". The requirements are ANDed.
+ type: object
+ strategy:
+ default: Any
+ description: 'strategy specifies the strategy to select the
+ target pod when multiple pods are selected. Valid values
+ are: - All: select all pods that match the labelsSelector.
+ - Any: select any one pod that match the labelsSelector.'
+ enum:
+ - All
+ - Any
type: string
- enable:
- description: enable or disable the schedule.
- type: boolean
- required:
- - cronExpression
- - enable
type: object
- startingDeadlineMinutes:
- description: startingDeadlineMinutes defines the deadline in minutes
- for starting the backup job if it misses scheduled time for
- any reason.
- format: int64
- maximum: 1440
- minimum: 0
- type: integer
- type: object
- snapshot:
- description: the policy for snapshot backup.
- properties:
- backupStatusUpdates:
- description: define how to update metadata for backup status.
- items:
- properties:
- containerName:
- description: which container name that kubectl can execute.
- type: string
- path:
- description: 'specify the json path of backup object for
- patch. example: manifests.backupLog -- means patch the
- backup json path of status.manifests.backupLog.'
- type: string
- script:
- description: the shell Script commands to collect backup
- status metadata. The script must exist in the container
- of ContainerName and the output format must be set to
- JSON. Note that outputting to stderr may cause the result
- format to not be in JSON.
- type: string
- updateStage:
- description: 'when to update the backup status, pre: before
- backup, post: after backup'
- enum:
- - pre
- - post
- type: string
- useTargetPodServiceAccount:
- description: useTargetPodServiceAccount defines whether
- this job requires the service account of the backup target
- pod. if true, will use the service account of the backup
- target pod. otherwise, will use the system service account.
- type: boolean
- required:
- - updateStage
- type: object
- type: array
- backupsHistoryLimit:
- default: 7
- description: the number of automatic backups to retain. Value
- must be non-negative integer. 0 means NO limit on the number
- of backups.
- format: int32
- type: integer
- hooks:
- description: execute hook commands for backup.
+ x-kubernetes-map-type: atomic
+ resources:
+ description: resources specifies the kubernetes resources to back
+ up.
properties:
- containerName:
- description: which container can exec command
- type: string
- image:
- description: exec command with image
- type: string
- postCommands:
- description: post backup to perform commands
+ excluded:
+ description: excluded is a slice of namespaced-scoped resource
+ type names to exclude in the kubernetes resources. The default
+ value is empty.
items:
type: string
type: array
- preCommands:
- description: pre backup to perform commands
+ included:
+ default:
+ - '*'
+ description: included is a slice of namespaced-scoped resource
+ type names to include in the kubernetes resources. The default
+ value is "*", which means all resource types will be included.
items:
type: string
type: array
- type: object
- onFailAttempted:
- description: count of backup stop retries on fail.
- format: int32
- type: integer
- target:
- description: target database cluster for backup.
- properties:
- labelsSelector:
- description: labelsSelector is used to find matching pods.
- Pods that match this label selector are counted to determine
- the number of pods in their corresponding topology domain.
+ selector:
+ description: selector is a metav1.LabelSelector to filter
+ the target kubernetes resources that need to be backed up.
+ If not set, will do not back up any kubernetes resources.
properties:
matchExpressions:
description: matchExpressions is a list of label selector
@@ -628,64 +475,35 @@ spec:
only "value". The requirements are ANDed.
type: object
type: object
- x-kubernetes-preserve-unknown-fields: true
- secret:
- description: secret is used to connect to the target database
- cluster. If not set, secret will be inherited from backup
- policy template. if still not set, the controller will check
- if any system account for dataprotection has been created.
- properties:
- name:
- description: the secret name
- pattern: ^[a-z0-9]([a-z0-9\.\-]*[a-z0-9])?$
- type: string
- passwordKey:
- default: password
- description: passwordKey the map key of the password in
- the connection credential secret
- type: string
- usernameKey:
- default: username
- description: usernameKey the map key of the user in the
- connection credential secret
- type: string
- required:
- - name
- type: object
- required:
- - labelsSelector
+ x-kubernetes-map-type: atomic
type: object
- required:
- - target
+ serviceAccountName:
+ description: serviceAccountName specifies the service account
+ to run the backup workload.
+ type: string
type: object
+ required:
+ - backupMethods
+ - target
type: object
status:
description: BackupPolicyStatus defines the observed state of BackupPolicy
properties:
- failureReason:
- description: the reason if backup policy check failed.
- type: string
- lastScheduleTime:
- description: information when was the last time the job was successfully
- scheduled.
- format: date-time
- type: string
- lastSuccessfulTime:
- description: information when was the last time the job successfully
- completed.
- format: date-time
+ message:
+ description: A human-readable message indicating details about why
+ the BackupPolicy is in this phase.
type: string
observedGeneration:
description: observedGeneration is the most recent generation observed
- for this BackupPolicy. It corresponds to the Cluster's generation,
+ for this BackupPolicy. It refers to the BackupPolicy's generation,
which is updated on mutation by the API Server.
format: int64
type: integer
phase:
- description: 'backup policy phase valid value: Available, Failed.'
+ description: phase - in list of [Available,Unavailable]
enum:
- Available
- - Failed
+ - Unavailable
type: string
type: object
type: object
diff --git a/config/crd/bases/dataprotection.kubeblocks.io_backuprepos.yaml b/config/crd/bases/dataprotection.kubeblocks.io_backuprepos.yaml
index 4cb9aea608e..00fcc2ab7a4 100644
--- a/config/crd/bases/dataprotection.kubeblocks.io_backuprepos.yaml
+++ b/config/crd/bases/dataprotection.kubeblocks.io_backuprepos.yaml
@@ -2,8 +2,7 @@ apiVersion: apiextensions.k8s.io/v1
kind: CustomResourceDefinition
metadata:
annotations:
- controller-gen.kubebuilder.io/version: v0.9.0
- creationTimestamp: null
+ controller-gen.kubebuilder.io/version: v0.12.1
labels:
app.kubernetes.io/name: kubeblocks
name: backuprepos.dataprotection.kubeblocks.io
@@ -51,6 +50,13 @@ spec:
spec:
description: BackupRepoSpec defines the desired state of BackupRepo
properties:
+ accessMethod:
+ default: Mount
+ description: Specifies the access method of the backup repo.
+ enum:
+ - Mount
+ - Tool
+ type: string
config:
additionalProperties:
type: string
@@ -69,6 +75,7 @@ spec:
name must be unique.
type: string
type: object
+ x-kubernetes-map-type: atomic
pvReclaimPolicy:
description: The reclaim policy for the PV created by this backup
repo.
@@ -183,6 +190,7 @@ spec:
name must be unique.
type: string
type: object
+ x-kubernetes-map-type: atomic
generatedStorageClassName:
description: generatedStorageClassName indicates the generated storage
class name.
@@ -211,6 +219,10 @@ spec:
description: Backup repo reconciliation phases. Valid values are PreChecking,
Failed, Ready, Deleting.
type: string
+ toolConfigSecretName:
+ description: toolConfigSecretName is the name of the secret containing
+ the configuration for the access tool.
+ type: string
type: object
type: object
served: true
diff --git a/config/crd/bases/dataprotection.kubeblocks.io_backups.yaml b/config/crd/bases/dataprotection.kubeblocks.io_backups.yaml
index 3bce53724b5..0126b52ced5 100644
--- a/config/crd/bases/dataprotection.kubeblocks.io_backups.yaml
+++ b/config/crd/bases/dataprotection.kubeblocks.io_backups.yaml
@@ -2,8 +2,7 @@ apiVersion: apiextensions.k8s.io/v1
kind: CustomResourceDefinition
metadata:
annotations:
- controller-gen.kubebuilder.io/version: v0.9.0
- creationTimestamp: null
+ controller-gen.kubebuilder.io/version: v0.12.1
labels:
app.kubernetes.io/name: kubeblocks
name: backups.dataprotection.kubeblocks.io
@@ -19,15 +18,18 @@ spec:
scope: Namespaced
versions:
- additionalPrinterColumns:
- - jsonPath: .spec.backupType
- name: TYPE
+ - jsonPath: .spec.backupPolicyName
+ name: POLICY
+ type: string
+ - jsonPath: .spec.backupMethod
+ name: METHOD
+ type: string
+ - jsonPath: .status.backupRepoName
+ name: REPO
type: string
- jsonPath: .status.phase
name: STATUS
type: string
- - jsonPath: .status.sourceCluster
- name: SOURCE-CLUSTER
- type: string
- jsonPath: .status.totalSize
name: TOTAL-SIZE
type: string
@@ -35,15 +37,18 @@ spec:
name: DURATION
type: string
- jsonPath: .metadata.creationTimestamp
- name: CREATE-TIME
+ name: CREATION-TIME
type: string
- jsonPath: .status.completionTimestamp
name: COMPLETION-TIME
type: string
+ - jsonPath: .status.expiration
+ name: EXPIRATION-TIME
+ type: string
name: v1alpha1
schema:
openAPIV3Schema:
- description: Backup is the Schema for the backups API (defined by User).
+ description: Backup is the Schema for the backups API.
properties:
apiVersion:
description: 'APIVersion defines the versioned schema of this representation
@@ -60,135 +65,444 @@ spec:
spec:
description: BackupSpec defines the desired state of Backup.
properties:
+ backupMethod:
+ description: backupMethod specifies the backup method name that is
+ defined in backupPolicy.
+ type: string
backupPolicyName:
- description: Which backupPolicy is applied to perform this backup
+ description: Which backupPolicy is applied to perform this backup.
pattern: ^[a-z0-9]([a-z0-9\.\-]*[a-z0-9])?$
type: string
- backupType:
- default: datafile
- description: Backup Type. datafile or logfile or snapshot. If not
- set, datafile is the default type.
- enum:
- - datafile
- - logfile
- - snapshot
+ deletionPolicy:
+ allOf:
+ - enum:
+ - Delete
+ - Retain
+ - enum:
+ - Delete
+ - Retain
+ default: Delete
+ description: deletionPolicy determines whether the backup contents
+ stored in backup repository should be deleted when the backup custom
+ resource is deleted. Supported values are "Retain" and "Delete".
+ "Retain" means that the backup content and its physical snapshot
+ on backup repository are kept. "Delete" means that the backup content
+ and its physical snapshot on backup repository are deleted.
type: string
parentBackupName:
- description: if backupType is incremental, parentBackupName is required.
+ description: parentBackupName determines the parent backup name for
+ incremental or differential backup.
+ type: string
+ retentionPeriod:
+ default: 7d
+ description: "retentionPeriod determines a duration up to which the
+ backup should be kept. controller will remove all backups that are
+ older than the RetentionPeriod. For example, RetentionPeriod of
+ `30d` will keep only the backups of last 30 days. Sample duration
+ format: - years: \t2y - months: \t6mo - days: \t\t30d - hours: \t12h
+ - minutes: \t30m You can also combine the above durations. For example:
+ 30d12h30m"
type: string
required:
+ - backupMethod
- backupPolicyName
- - backupType
type: object
status:
description: BackupStatus defines the observed state of Backup.
properties:
- availableReplicas:
- description: availableReplicas available replicas for statefulSet
- which created by backup.
- format: int32
- type: integer
- backupToolName:
- description: backupToolName references the backup tool name.
+ actions:
+ description: actions records the actions information for this backup.
+ items:
+ properties:
+ actionType:
+ description: actionType is the type of the action.
+ type: string
+ availableReplicas:
+ description: availableReplicas available replicas for statefulSet
+ action.
+ format: int32
+ type: integer
+ completionTimestamp:
+ description: completionTimestamp records the time an action
+ was completed.
+ format: date-time
+ type: string
+ failureReason:
+ description: failureReason is an error that caused the backup
+ to fail.
+ type: string
+ name:
+ description: name is the name of the action.
+ type: string
+ objectRef:
+ description: objectRef is the object reference for the action.
+ properties:
+ apiVersion:
+ description: API version of the referent.
+ type: string
+ fieldPath:
+ description: 'If referring to a piece of an object instead
+ of an entire object, this string should contain a valid
+ JSON/Go field access statement, such as desiredState.manifest.containers[2].
+ For example, if the object reference is to a container
+ within a pod, this would take on a value like: "spec.containers{name}"
+ (where "name" refers to the name of the container that
+ triggered the event) or if no container name is specified
+ "spec.containers[2]" (container with index 2 in this pod).
+ This syntax is chosen only to have some well-defined way
+ of referencing a part of an object. TODO: this design
+ is not final and this field is subject to change in the
+ future.'
+ type: string
+ kind:
+ description: 'Kind of the referent. More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#types-kinds'
+ type: string
+ name:
+ description: 'Name of the referent. More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names'
+ type: string
+ namespace:
+ description: 'Namespace of the referent. More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/namespaces/'
+ type: string
+ resourceVersion:
+ description: 'Specific resourceVersion to which this reference
+ is made, if any. More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#concurrency-control-and-consistency'
+ type: string
+ uid:
+ description: 'UID of the referent. More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#uids'
+ type: string
+ type: object
+ x-kubernetes-map-type: atomic
+ phase:
+ description: phase is the current state of the action.
+ type: string
+ startTimestamp:
+ description: startTimestamp records the time an action was started.
+ format: date-time
+ type: string
+ timeRange:
+ description: timeRange records the time range of backed up data,
+ for PITR, this is the time range of recoverable data.
+ properties:
+ end:
+ description: end records the end time of backup.
+ format: date-time
+ type: string
+ start:
+ description: start records the start time of backup.
+ format: date-time
+ type: string
+ type: object
+ totalSize:
+ description: totalSize is the total size of backed up data size.
+ A string with capacity units in the format of "1Gi", "1Mi",
+ "1Ki".
+ type: string
+ volumeSnapshots:
+ description: volumeSnapshots records the volume snapshot status
+ for the action.
+ items:
+ properties:
+ contentName:
+ description: contentName is the name of the volume snapshot
+ content.
+ type: string
+ name:
+ description: name is the name of the volume snapshot.
+ type: string
+ size:
+ description: size is the size of the volume snapshot.
+ type: string
+ volumeName:
+ description: volumeName is the name of the volume.
+ type: string
+ type: object
+ type: array
+ type: object
+ type: array
+ backupMethod:
+ description: backupMethod records the backup method information for
+ this backup. Refer to BackupMethod for more details.
+ properties:
+ actionSetName:
+ description: actionSetName refers to the ActionSet object that
+ defines the backup actions. For volume snapshot backup, the
+ actionSet is not required, the controller will use the CSI volume
+ snapshotter to create the snapshot.
+ type: string
+ env:
+ description: env specifies the environment variables for the backup
+ workload.
+ items:
+ description: EnvVar represents an environment variable present
+ in a Container.
+ properties:
+ name:
+ description: Name of the environment variable. Must be a
+ C_IDENTIFIER.
+ type: string
+ value:
+ description: 'Variable references $(VAR_NAME) are expanded
+ using the previously defined environment variables in
+ the container and any service environment variables. If
+ a variable cannot be resolved, the reference in the input
+ string will be unchanged. Double $$ are reduced to a single
+ $, which allows for escaping the $(VAR_NAME) syntax: i.e.
+ "$$(VAR_NAME)" will produce the string literal "$(VAR_NAME)".
+ Escaped references will never be expanded, regardless
+ of whether the variable exists or not. Defaults to "".'
+ type: string
+ valueFrom:
+ description: Source for the environment variable's value.
+ Cannot be used if value is not empty.
+ properties:
+ configMapKeyRef:
+ description: Selects a key of a ConfigMap.
+ properties:
+ key:
+ description: The key to select.
+ type: string
+ name:
+ description: 'Name of the referent. More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names
+ TODO: Add other useful fields. apiVersion, kind,
+ uid?'
+ type: string
+ optional:
+ description: Specify whether the ConfigMap or its
+ key must be defined
+ type: boolean
+ required:
+ - key
+ type: object
+ x-kubernetes-map-type: atomic
+ fieldRef:
+ description: 'Selects a field of the pod: supports metadata.name,
+ metadata.namespace, `metadata.labels['''']`,
+ `metadata.annotations['''']`, spec.nodeName,
+ spec.serviceAccountName, status.hostIP, status.podIP,
+ status.podIPs.'
+ properties:
+ apiVersion:
+ description: Version of the schema the FieldPath
+ is written in terms of, defaults to "v1".
+ type: string
+ fieldPath:
+ description: Path of the field to select in the
+ specified API version.
+ type: string
+ required:
+ - fieldPath
+ type: object
+ x-kubernetes-map-type: atomic
+ resourceFieldRef:
+ description: 'Selects a resource of the container: only
+ resources limits and requests (limits.cpu, limits.memory,
+ limits.ephemeral-storage, requests.cpu, requests.memory
+ and requests.ephemeral-storage) are currently supported.'
+ properties:
+ containerName:
+ description: 'Container name: required for volumes,
+ optional for env vars'
+ type: string
+ divisor:
+ anyOf:
+ - type: integer
+ - type: string
+ description: Specifies the output format of the
+ exposed resources, defaults to "1"
+ pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$
+ x-kubernetes-int-or-string: true
+ resource:
+ description: 'Required: resource to select'
+ type: string
+ required:
+ - resource
+ type: object
+ x-kubernetes-map-type: atomic
+ secretKeyRef:
+ description: Selects a key of a secret in the pod's
+ namespace
+ properties:
+ key:
+ description: The key of the secret to select from. Must
+ be a valid secret key.
+ type: string
+ name:
+ description: 'Name of the referent. More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names
+ TODO: Add other useful fields. apiVersion, kind,
+ uid?'
+ type: string
+ optional:
+ description: Specify whether the Secret or its key
+ must be defined
+ type: boolean
+ required:
+ - key
+ type: object
+ x-kubernetes-map-type: atomic
+ type: object
+ required:
+ - name
+ type: object
+ type: array
+ name:
+ description: the name of backup method.
+ pattern: ^[a-z0-9]([a-z0-9\.\-]*[a-z0-9])?$
+ type: string
+ runtimeSettings:
+ description: runtimeSettings specifies runtime settings for the
+ backup workload container.
+ properties:
+ resources:
+ description: 'resources specifies the resource required by
+ container. More info: https://kubernetes.io/docs/concepts/configuration/manage-compute-resources-container/'
+ properties:
+ claims:
+ description: "Claims lists the names of resources, defined
+ in spec.resourceClaims, that are used by this container.
+ \n This is an alpha field and requires enabling the
+ DynamicResourceAllocation feature gate. \n This field
+ is immutable. It can only be set for containers."
+ items:
+ description: ResourceClaim references one entry in PodSpec.ResourceClaims.
+ properties:
+ name:
+ description: Name must match the name of one entry
+ in pod.spec.resourceClaims of the Pod where this
+ field is used. It makes that resource available
+ inside a container.
+ type: string
+ required:
+ - name
+ type: object
+ type: array
+ x-kubernetes-list-map-keys:
+ - name
+ x-kubernetes-list-type: map
+ limits:
+ additionalProperties:
+ anyOf:
+ - type: integer
+ - type: string
+ pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$
+ x-kubernetes-int-or-string: true
+ description: 'Limits describes the maximum amount of compute
+ resources allowed. More info: https://kubernetes.io/docs/concepts/configuration/manage-resources-containers/'
+ type: object
+ requests:
+ additionalProperties:
+ anyOf:
+ - type: integer
+ - type: string
+ pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$
+ x-kubernetes-int-or-string: true
+ description: 'Requests describes the minimum amount of
+ compute resources required. If Requests is omitted for
+ a container, it defaults to Limits if that is explicitly
+ specified, otherwise to an implementation-defined value.
+ Requests cannot exceed Limits. More info: https://kubernetes.io/docs/concepts/configuration/manage-resources-containers/'
+ type: object
+ type: object
+ type: object
+ snapshotVolumes:
+ default: false
+ description: snapshotVolumes specifies whether to take snapshots
+ of persistent volumes. if true, the BackupScript is not required,
+ the controller will use the CSI volume snapshotter to create
+ the snapshot.
+ type: boolean
+ targetVolumes:
+ description: targetVolumes specifies which volumes from the target
+ should be mounted in the backup workload.
+ properties:
+ volumeMounts:
+ description: volumeMounts specifies the mount for the volumes
+ specified in `Volumes` section.
+ items:
+ description: VolumeMount describes a mounting of a Volume
+ within a container.
+ properties:
+ mountPath:
+ description: Path within the container at which the
+ volume should be mounted. Must not contain ':'.
+ type: string
+ mountPropagation:
+ description: mountPropagation determines how mounts
+ are propagated from the host to container and the
+ other way around. When not set, MountPropagationNone
+ is used. This field is beta in 1.10.
+ type: string
+ name:
+ description: This must match the Name of a Volume.
+ type: string
+ readOnly:
+ description: Mounted read-only if true, read-write otherwise
+ (false or unspecified). Defaults to false.
+ type: boolean
+ subPath:
+ description: Path within the volume from which the container's
+ volume should be mounted. Defaults to "" (volume's
+ root).
+ type: string
+ subPathExpr:
+ description: Expanded path within the volume from which
+ the container's volume should be mounted. Behaves
+ similarly to SubPath but environment variable references
+ $(VAR_NAME) are expanded using the container's environment.
+ Defaults to "" (volume's root). SubPathExpr and SubPath
+ are mutually exclusive.
+ type: string
+ required:
+ - mountPath
+ - name
+ type: object
+ type: array
+ volumes:
+ description: Volumes indicates the list of volumes of targeted
+ application that should be mounted on the backup job.
+ items:
+ type: string
+ type: array
+ type: object
+ required:
+ - name
+ type: object
+ backupRepoName:
+ description: backupRepoName is the name of the backup repository.
type: string
completionTimestamp:
- description: Date/time when the backup finished being processed.
+ description: completionTimestamp records the time a backup was completed.
+ Completion time is recorded even on failed backups. The server's
+ time is used for CompletionTimestamp.
format: date-time
type: string
duration:
description: The duration time of backup execution. When converted
- to a string, the form is "1h2m0.5s".
+ to a string, the format is "1h2m0.5s".
type: string
expiration:
- description: The date and time when the Backup is eligible for garbage
- collection. 'null' means the Backup is NOT be cleaned except delete
+ description: expiration is when this backup is eligible for garbage
+ collection. 'null' means the Backup will NOT be cleaned except delete
manual.
format: date-time
type: string
failureReason:
- description: The reason for a backup failure.
+ description: failureReason is an error that caused the backup to fail.
type: string
- logFilePersistentVolumeClaimName:
- description: logFilePersistentVolumeClaimName saves the logfile backup
- data.
+ formatVersion:
+ description: formatVersion is the backup format version, including
+ major, minor and patch version.
type: string
- manifests:
- description: manifests determines the backup metadata info.
- properties:
- backupLog:
- description: backupLog records startTime and stopTime of data
- logging.
- properties:
- startTime:
- description: startTime records the start time of data logging.
- format: date-time
- type: string
- stopTime:
- description: stopTime records the stop time of data logging.
- format: date-time
- type: string
- type: object
- backupSnapshot:
- description: snapshot records the volume snapshot metadata.
- properties:
- volumeSnapshotContentName:
- description: volumeSnapshotContentName specifies the name
- of a pre-existing VolumeSnapshotContent object representing
- an existing volume snapshot. This field should be set if
- the snapshot already exists and only needs a representation
- in Kubernetes. This field is immutable.
- type: string
- volumeSnapshotName:
- description: volumeSnapshotName records the volumeSnapshot
- name.
- type: string
- type: object
- backupTool:
- description: backupTool records information about backup files
- generated by the backup tool.
- properties:
- checkpoint:
- description: backup checkpoint, for incremental backup.
- type: string
- checksum:
- description: checksum of backup file, generated by md5 or
- sha1 or sha256.
- type: string
- filePath:
- description: filePath records the file path of backup.
- type: string
- logFilePath:
- description: logFilePath records the log file path of backup.
- type: string
- uploadTotalSize:
- description: Backup upload total size. A string with capacity
- units in the form of "1Gi", "1Mi", "1Ki".
- type: string
- volumeName:
- description: volumeName records volume name of backup data
- pvc.
- type: string
- type: object
- target:
- description: target records the target cluster metadata string,
- which is in JSON format.
- type: string
- userContext:
- additionalProperties:
- type: string
- description: userContext stores some loosely structured and extensible
- information.
- type: object
- type: object
- parentBackupName:
- description: Records parentBackupName if backupType is incremental.
+ path:
+ description: path is the directory inside the backup repository where
+ the backup data is stored. It is an absolute path in the backup
+ repository.
type: string
persistentVolumeClaimName:
- description: remoteVolume saves the backup data.
+ description: persistentVolumeClaimName is the name of the persistent
+ volume claim that is used to store the backup data.
type: string
phase:
- description: BackupPhase The current phase. Valid values are New,
- InProgress, Completed, Failed.
+ description: phase is the current state of the Backup.
enum:
- New
- InProgress
@@ -197,18 +511,209 @@ spec:
- Failed
- Deleting
type: string
- sourceCluster:
- description: sourceCluster records the source cluster information
- for this backup.
- type: string
startTimestamp:
- description: Date/time when the backup started being processed.
+ description: startTimestamp records the time a backup was started.
+ The server's time is used for StartTimestamp.
format: date-time
type: string
+ target:
+ description: target records the target information for this backup.
+ properties:
+ connectionCredential:
+ description: connectionCredential specifies the connection credential
+ to connect to the target database cluster.
+ properties:
+ hostKey:
+ description: hostKey specifies the map key of the host in
+ the connection credential secret.
+ type: string
+ passwordKey:
+ default: password
+ description: passwordKey specifies the map key of the password
+ in the connection credential secret.
+ type: string
+ portKey:
+ default: port
+ description: portKey specifies the map key of the port in
+ the connection credential secret.
+ type: string
+ secretName:
+ description: secretName refers to the Secret object that contains
+ the connection credential.
+ type: string
+ usernameKey:
+ default: username
+ description: usernameKey specifies the map key of the user
+ in the connection credential secret.
+ type: string
+ type: object
+ podSelector:
+ description: podSelector is used to find the target pod. The volumes
+ of the target pod will be backed up.
+ properties:
+ matchExpressions:
+ description: matchExpressions is a list of label selector
+ requirements. The requirements are ANDed.
+ items:
+ description: A label selector requirement is a selector
+ that contains values, a key, and an operator that relates
+ the key and values.
+ properties:
+ key:
+ description: key is the label key that the selector
+ applies to.
+ type: string
+ operator:
+ description: operator represents a key's relationship
+ to a set of values. Valid operators are In, NotIn,
+ Exists and DoesNotExist.
+ type: string
+ values:
+ description: values is an array of string values. If
+ the operator is In or NotIn, the values array must
+ be non-empty. If the operator is Exists or DoesNotExist,
+ the values array must be empty. This array is replaced
+ during a strategic merge patch.
+ items:
+ type: string
+ type: array
+ required:
+ - key
+ - operator
+ type: object
+ type: array
+ matchLabels:
+ additionalProperties:
+ type: string
+ description: matchLabels is a map of {key,value} pairs. A
+ single {key,value} in the matchLabels map is equivalent
+ to an element of matchExpressions, whose key field is "key",
+ the operator is "In", and the values array contains only
+ "value". The requirements are ANDed.
+ type: object
+ strategy:
+ default: Any
+ description: 'strategy specifies the strategy to select the
+ target pod when multiple pods are selected. Valid values
+ are: - All: select all pods that match the labelsSelector.
+ - Any: select any one pod that match the labelsSelector.'
+ enum:
+ - All
+ - Any
+ type: string
+ type: object
+ x-kubernetes-map-type: atomic
+ resources:
+ description: resources specifies the kubernetes resources to back
+ up.
+ properties:
+ excluded:
+ description: excluded is a slice of namespaced-scoped resource
+ type names to exclude in the kubernetes resources. The default
+ value is empty.
+ items:
+ type: string
+ type: array
+ included:
+ default:
+ - '*'
+ description: included is a slice of namespaced-scoped resource
+ type names to include in the kubernetes resources. The default
+ value is "*", which means all resource types will be included.
+ items:
+ type: string
+ type: array
+ selector:
+ description: selector is a metav1.LabelSelector to filter
+ the target kubernetes resources that need to be backed up.
+ If not set, will do not back up any kubernetes resources.
+ properties:
+ matchExpressions:
+ description: matchExpressions is a list of label selector
+ requirements. The requirements are ANDed.
+ items:
+ description: A label selector requirement is a selector
+ that contains values, a key, and an operator that
+ relates the key and values.
+ properties:
+ key:
+ description: key is the label key that the selector
+ applies to.
+ type: string
+ operator:
+ description: operator represents a key's relationship
+ to a set of values. Valid operators are In, NotIn,
+ Exists and DoesNotExist.
+ type: string
+ values:
+ description: values is an array of string values.
+ If the operator is In or NotIn, the values array
+ must be non-empty. If the operator is Exists or
+ DoesNotExist, the values array must be empty.
+ This array is replaced during a strategic merge
+ patch.
+ items:
+ type: string
+ type: array
+ required:
+ - key
+ - operator
+ type: object
+ type: array
+ matchLabels:
+ additionalProperties:
+ type: string
+ description: matchLabels is a map of {key,value} pairs.
+ A single {key,value} in the matchLabels map is equivalent
+ to an element of matchExpressions, whose key field is
+ "key", the operator is "In", and the values array contains
+ only "value". The requirements are ANDed.
+ type: object
+ type: object
+ x-kubernetes-map-type: atomic
+ type: object
+ serviceAccountName:
+ description: serviceAccountName specifies the service account
+ to run the backup workload.
+ type: string
+ type: object
+ timeRange:
+ description: timeRange records the time range of backed up data, for
+ PITR, this is the time range of recoverable data.
+ properties:
+ end:
+ description: end records the end time of backup.
+ format: date-time
+ type: string
+ start:
+ description: start records the start time of backup.
+ format: date-time
+ type: string
+ type: object
totalSize:
- description: Backup total size. A string with capacity units in the
- form of "1Gi", "1Mi", "1Ki".
+ description: totalSize is the total size of backed up data size. A
+ string with capacity units in the format of "1Gi", "1Mi", "1Ki".
type: string
+ volumeSnapshots:
+ description: volumeSnapshots records the volume snapshot status for
+ the action.
+ items:
+ properties:
+ contentName:
+ description: contentName is the name of the volume snapshot
+ content.
+ type: string
+ name:
+ description: name is the name of the volume snapshot.
+ type: string
+ size:
+ description: size is the size of the volume snapshot.
+ type: string
+ volumeName:
+ description: volumeName is the name of the volume.
+ type: string
+ type: object
+ type: array
type: object
type: object
served: true
diff --git a/config/crd/bases/dataprotection.kubeblocks.io_backupschedules.yaml b/config/crd/bases/dataprotection.kubeblocks.io_backupschedules.yaml
new file mode 100644
index 00000000000..40d07aa5fc7
--- /dev/null
+++ b/config/crd/bases/dataprotection.kubeblocks.io_backupschedules.yaml
@@ -0,0 +1,141 @@
+apiVersion: apiextensions.k8s.io/v1
+kind: CustomResourceDefinition
+metadata:
+ annotations:
+ controller-gen.kubebuilder.io/version: v0.12.1
+ labels:
+ app.kubernetes.io/name: kubeblocks
+ name: backupschedules.dataprotection.kubeblocks.io
+spec:
+ group: dataprotection.kubeblocks.io
+ names:
+ categories:
+ - kubeblocks
+ kind: BackupSchedule
+ listKind: BackupScheduleList
+ plural: backupschedules
+ shortNames:
+ - bs
+ singular: backupschedule
+ scope: Namespaced
+ versions:
+ - additionalPrinterColumns:
+ - jsonPath: .status.phase
+ name: STATUS
+ type: string
+ - jsonPath: .metadata.creationTimestamp
+ name: AGE
+ type: date
+ name: v1alpha1
+ schema:
+ openAPIV3Schema:
+ description: BackupSchedule is the Schema for the backupschedules API.
+ properties:
+ apiVersion:
+ description: 'APIVersion defines the versioned schema of this representation
+ of an object. Servers should convert recognized schemas to the latest
+ internal value, and may reject unrecognized values. More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#resources'
+ type: string
+ kind:
+ description: 'Kind is a string value representing the REST resource this
+ object represents. Servers may infer this from the endpoint the client
+ submits requests to. Cannot be updated. In CamelCase. More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#types-kinds'
+ type: string
+ metadata:
+ type: object
+ spec:
+ description: BackupScheduleSpec defines the desired state of BackupSchedule.
+ properties:
+ backupPolicyName:
+ description: Which backupPolicy is applied to perform this backup.
+ pattern: ^[a-z0-9]([a-z0-9\.\-]*[a-z0-9])?$
+ type: string
+ schedules:
+ description: schedules defines the list of backup schedules.
+ items:
+ properties:
+ backupMethod:
+ description: backupMethod specifies the backup method name that
+ is defined in backupPolicy.
+ type: string
+ cronExpression:
+ description: the cron expression for schedule, the timezone
+ is in UTC. see https://en.wikipedia.org/wiki/Cron.
+ type: string
+ enabled:
+ description: enabled specifies whether the backup schedule is
+ enabled or not.
+ type: boolean
+ retentionPeriod:
+ default: 7d
+ description: "retentionPeriod determines a duration up to which
+ the backup should be kept. controller will remove all backups
+ that are older than the RetentionPeriod. For example, RetentionPeriod
+ of `30d` will keep only the backups of last 30 days. Sample
+ duration format: - years: \t2y - months: \t6mo - days: \t\t30d
+ - hours: \t12h - minutes: \t30m You can also combine the above
+ durations. For example: 30d12h30m"
+ type: string
+ required:
+ - backupMethod
+ - cronExpression
+ type: object
+ minItems: 1
+ type: array
+ startingDeadlineMinutes:
+ description: startingDeadlineMinutes defines the deadline in minutes
+ for starting the backup workload if it misses scheduled time for
+ any reason.
+ format: int64
+ maximum: 1440
+ minimum: 0
+ type: integer
+ required:
+ - backupPolicyName
+ - schedules
+ type: object
+ status:
+ description: BackupScheduleStatus defines the observed state of BackupSchedule.
+ properties:
+ failureReason:
+ description: failureReason is an error that caused the backup to fail.
+ type: string
+ observedGeneration:
+ description: observedGeneration is the most recent generation observed
+ for this BackupSchedule. It refers to the BackupSchedule's generation,
+ which is updated on mutation by the API Server.
+ format: int64
+ type: integer
+ phase:
+ description: phase describes the phase of the BackupSchedule.
+ type: string
+ schedules:
+ additionalProperties:
+ description: ScheduleStatus defines the status of each schedule.
+ properties:
+ failureReason:
+ description: failureReason is an error that caused the backup
+ to fail.
+ type: string
+ lastScheduleTime:
+ description: lastScheduleTime records the last time the backup
+ was scheduled.
+ format: date-time
+ type: string
+ lastSuccessfulTime:
+ description: lastSuccessfulTime records the last time the backup
+ was successfully completed.
+ format: date-time
+ type: string
+ phase:
+ description: phase describes the phase of the schedule.
+ type: string
+ type: object
+ description: schedules describes the status of each schedule.
+ type: object
+ type: object
+ type: object
+ served: true
+ storage: true
+ subresources:
+ status: {}
diff --git a/config/crd/bases/dataprotection.kubeblocks.io_backuptools.yaml b/config/crd/bases/dataprotection.kubeblocks.io_backuptools.yaml
deleted file mode 100644
index 2bec3e71768..00000000000
--- a/config/crd/bases/dataprotection.kubeblocks.io_backuptools.yaml
+++ /dev/null
@@ -1,324 +0,0 @@
-apiVersion: apiextensions.k8s.io/v1
-kind: CustomResourceDefinition
-metadata:
- annotations:
- controller-gen.kubebuilder.io/version: v0.9.0
- creationTimestamp: null
- labels:
- app.kubernetes.io/name: kubeblocks
- name: backuptools.dataprotection.kubeblocks.io
-spec:
- group: dataprotection.kubeblocks.io
- names:
- categories:
- - kubeblocks
- kind: BackupTool
- listKind: BackupToolList
- plural: backuptools
- singular: backuptool
- scope: Cluster
- versions:
- - name: v1alpha1
- schema:
- openAPIV3Schema:
- description: BackupTool is the Schema for the backuptools API (defined by
- provider)
- properties:
- apiVersion:
- description: 'APIVersion defines the versioned schema of this representation
- of an object. Servers should convert recognized schemas to the latest
- internal value, and may reject unrecognized values. More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#resources'
- type: string
- kind:
- description: 'Kind is a string value representing the REST resource this
- object represents. Servers may infer this from the endpoint the client
- submits requests to. Cannot be updated. In CamelCase. More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#types-kinds'
- type: string
- metadata:
- type: object
- spec:
- description: BackupToolSpec defines the desired state of BackupTool
- properties:
- backupCommands:
- description: Array of command that apps can do database backup. from
- invoke args the order of commands follows the order of array.
- items:
- type: string
- type: array
- deployKind:
- default: job
- description: 'which kind for run a backup tool, supported values:
- job, statefulSet.'
- enum:
- - job
- - statefulSet
- type: string
- env:
- description: List of environment variables to set in the container.
- items:
- description: EnvVar represents an environment variable present in
- a Container.
- properties:
- name:
- description: Name of the environment variable. Must be a C_IDENTIFIER.
- type: string
- value:
- description: 'Variable references $(VAR_NAME) are expanded using
- the previously defined environment variables in the container
- and any service environment variables. If a variable cannot
- be resolved, the reference in the input string will be unchanged.
- Double $$ are reduced to a single $, which allows for escaping
- the $(VAR_NAME) syntax: i.e. "$$(VAR_NAME)" will produce the
- string literal "$(VAR_NAME)". Escaped references will never
- be expanded, regardless of whether the variable exists or
- not. Defaults to "".'
- type: string
- valueFrom:
- description: Source for the environment variable's value. Cannot
- be used if value is not empty.
- properties:
- configMapKeyRef:
- description: Selects a key of a ConfigMap.
- properties:
- key:
- description: The key to select.
- type: string
- name:
- description: 'Name of the referent. More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names
- TODO: Add other useful fields. apiVersion, kind, uid?'
- type: string
- optional:
- description: Specify whether the ConfigMap or its key
- must be defined
- type: boolean
- required:
- - key
- type: object
- fieldRef:
- description: 'Selects a field of the pod: supports metadata.name,
- metadata.namespace, `metadata.labels['''']`, `metadata.annotations['''']`,
- spec.nodeName, spec.serviceAccountName, status.hostIP,
- status.podIP, status.podIPs.'
- properties:
- apiVersion:
- description: Version of the schema the FieldPath is
- written in terms of, defaults to "v1".
- type: string
- fieldPath:
- description: Path of the field to select in the specified
- API version.
- type: string
- required:
- - fieldPath
- type: object
- resourceFieldRef:
- description: 'Selects a resource of the container: only
- resources limits and requests (limits.cpu, limits.memory,
- limits.ephemeral-storage, requests.cpu, requests.memory
- and requests.ephemeral-storage) are currently supported.'
- properties:
- containerName:
- description: 'Container name: required for volumes,
- optional for env vars'
- type: string
- divisor:
- anyOf:
- - type: integer
- - type: string
- description: Specifies the output format of the exposed
- resources, defaults to "1"
- pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$
- x-kubernetes-int-or-string: true
- resource:
- description: 'Required: resource to select'
- type: string
- required:
- - resource
- type: object
- secretKeyRef:
- description: Selects a key of a secret in the pod's namespace
- properties:
- key:
- description: The key of the secret to select from. Must
- be a valid secret key.
- type: string
- name:
- description: 'Name of the referent. More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names
- TODO: Add other useful fields. apiVersion, kind, uid?'
- type: string
- optional:
- description: Specify whether the Secret or its key must
- be defined
- type: boolean
- required:
- - key
- type: object
- type: object
- required:
- - name
- type: object
- type: array
- x-kubernetes-preserve-unknown-fields: true
- envFrom:
- description: List of sources to populate environment variables in
- the container. The keys defined within a source must be a C_IDENTIFIER.
- All invalid keys will be reported as an event when the container
- is starting. When a key exists in multiple sources, the value associated
- with the last source will take precedence. Values defined by an
- Env with a duplicate key will take precedence. Cannot be updated.
- items:
- description: EnvFromSource represents the source of a set of ConfigMaps
- properties:
- configMapRef:
- description: The ConfigMap to select from
- properties:
- name:
- description: 'Name of the referent. More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names
- TODO: Add other useful fields. apiVersion, kind, uid?'
- type: string
- optional:
- description: Specify whether the ConfigMap must be defined
- type: boolean
- type: object
- prefix:
- description: An optional identifier to prepend to each key in
- the ConfigMap. Must be a C_IDENTIFIER.
- type: string
- secretRef:
- description: The Secret to select from
- properties:
- name:
- description: 'Name of the referent. More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names
- TODO: Add other useful fields. apiVersion, kind, uid?'
- type: string
- optional:
- description: Specify whether the Secret must be defined
- type: boolean
- type: object
- type: object
- type: array
- x-kubernetes-preserve-unknown-fields: true
- image:
- description: Backup tool Container image name.
- type: string
- incrementalBackupCommands:
- description: Array of command that apps can do database incremental
- backup. like xtrabackup, that can performs an incremental backup
- file.
- items:
- type: string
- type: array
- logical:
- description: backup tool can support logical restore, in this case,
- restore NOT RESTART database.
- properties:
- incrementalRestoreCommands:
- description: Array of incremental restore commands.
- items:
- type: string
- type: array
- podScope:
- default: All
- description: 'podScope defines the pod scope for restore from
- backup, supported values: - ''All'' will exec the restore command
- on all pods. - ''ReadWrite'' will pick a ReadWrite pod to exec
- the restore command.'
- enum:
- - All
- - ReadWrite
- type: string
- restoreCommands:
- description: Array of command that apps can perform database restore.
- like xtrabackup, that can performs restore mysql from files.
- items:
- type: string
- type: array
- type: object
- physical:
- description: backup tool can support physical restore, in this case,
- restore must be RESTART database.
- properties:
- incrementalRestoreCommands:
- description: Array of incremental restore commands.
- items:
- type: string
- type: array
- relyOnLogfile:
- description: relyOnLogfile defines whether the current recovery
- relies on log files
- type: boolean
- restoreCommands:
- description: Array of command that apps can perform database restore.
- like xtrabackup, that can performs restore mysql from files.
- items:
- type: string
- type: array
- type: object
- resources:
- description: Compute Resources required by this container. Cannot
- be updated.
- properties:
- claims:
- description: "Claims lists the names of resources, defined in
- spec.resourceClaims, that are used by this container. \n This
- is an alpha field and requires enabling the DynamicResourceAllocation
- feature gate. \n This field is immutable. It can only be set
- for containers."
- items:
- description: ResourceClaim references one entry in PodSpec.ResourceClaims.
- properties:
- name:
- description: Name must match the name of one entry in pod.spec.resourceClaims
- of the Pod where this field is used. It makes that resource
- available inside a container.
- type: string
- required:
- - name
- type: object
- type: array
- x-kubernetes-list-map-keys:
- - name
- x-kubernetes-list-type: map
- limits:
- additionalProperties:
- anyOf:
- - type: integer
- - type: string
- pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$
- x-kubernetes-int-or-string: true
- description: 'Limits describes the maximum amount of compute resources
- allowed. More info: https://kubernetes.io/docs/concepts/configuration/manage-resources-containers/'
- type: object
- requests:
- additionalProperties:
- anyOf:
- - type: integer
- - type: string
- pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$
- x-kubernetes-int-or-string: true
- description: 'Requests describes the minimum amount of compute
- resources required. If Requests is omitted for a container,
- it defaults to Limits if that is explicitly specified, otherwise
- to an implementation-defined value. More info: https://kubernetes.io/docs/concepts/configuration/manage-resources-containers/'
- type: object
- type: object
- x-kubernetes-preserve-unknown-fields: true
- type:
- default: file
- description: the type of backup tool, file or pitr
- enum:
- - file
- - pitr
- type: string
- required:
- - backupCommands
- - image
- type: object
- status:
- description: BackupToolStatus defines the observed state of BackupTool
- type: object
- type: object
- served: true
- storage: true
- subresources:
- status: {}
diff --git a/config/crd/bases/dataprotection.kubeblocks.io_restorejobs.yaml b/config/crd/bases/dataprotection.kubeblocks.io_restorejobs.yaml
deleted file mode 100644
index 917f6a6390f..00000000000
--- a/config/crd/bases/dataprotection.kubeblocks.io_restorejobs.yaml
+++ /dev/null
@@ -1,1778 +0,0 @@
-apiVersion: apiextensions.k8s.io/v1
-kind: CustomResourceDefinition
-metadata:
- annotations:
- controller-gen.kubebuilder.io/version: v0.9.0
- creationTimestamp: null
- labels:
- app.kubernetes.io/name: kubeblocks
- name: restorejobs.dataprotection.kubeblocks.io
-spec:
- group: dataprotection.kubeblocks.io
- names:
- categories:
- - kubeblocks
- kind: RestoreJob
- listKind: RestoreJobList
- plural: restorejobs
- singular: restorejob
- scope: Namespaced
- versions:
- - additionalPrinterColumns:
- - jsonPath: .status.phase
- name: STATUS
- type: string
- - jsonPath: .status.completionTimestamp
- name: COMPLETION-TIME
- type: date
- - jsonPath: .metadata.creationTimestamp
- name: AGE
- type: date
- name: v1alpha1
- schema:
- openAPIV3Schema:
- description: RestoreJob is the Schema for the restorejobs API (defined by
- User)
- properties:
- apiVersion:
- description: 'APIVersion defines the versioned schema of this representation
- of an object. Servers should convert recognized schemas to the latest
- internal value, and may reject unrecognized values. More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#resources'
- type: string
- kind:
- description: 'Kind is a string value representing the REST resource this
- object represents. Servers may infer this from the endpoint the client
- submits requests to. Cannot be updated. In CamelCase. More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#types-kinds'
- type: string
- metadata:
- type: object
- spec:
- description: RestoreJobSpec defines the desired state of RestoreJob
- properties:
- backupJobName:
- description: Specified one backupJob to restore.
- type: string
- onFailAttempted:
- description: count of backup stop retries on fail.
- format: int32
- type: integer
- target:
- description: the target database workload to restore
- properties:
- labelsSelector:
- description: labelsSelector is used to find matching pods. Pods
- that match this label selector are counted to determine the
- number of pods in their corresponding topology domain.
- properties:
- matchExpressions:
- description: matchExpressions is a list of label selector
- requirements. The requirements are ANDed.
- items:
- description: A label selector requirement is a selector
- that contains values, a key, and an operator that relates
- the key and values.
- properties:
- key:
- description: key is the label key that the selector
- applies to.
- type: string
- operator:
- description: operator represents a key's relationship
- to a set of values. Valid operators are In, NotIn,
- Exists and DoesNotExist.
- type: string
- values:
- description: values is an array of string values. If
- the operator is In or NotIn, the values array must
- be non-empty. If the operator is Exists or DoesNotExist,
- the values array must be empty. This array is replaced
- during a strategic merge patch.
- items:
- type: string
- type: array
- required:
- - key
- - operator
- type: object
- type: array
- matchLabels:
- additionalProperties:
- type: string
- description: matchLabels is a map of {key,value} pairs. A
- single {key,value} in the matchLabels map is equivalent
- to an element of matchExpressions, whose key field is "key",
- the operator is "In", and the values array contains only
- "value". The requirements are ANDed.
- type: object
- type: object
- x-kubernetes-preserve-unknown-fields: true
- secret:
- description: secret is used to connect to the target database
- cluster. If not set, secret will be inherited from backup policy
- template. if still not set, the controller will check if any
- system account for dataprotection has been created.
- properties:
- name:
- description: the secret name
- pattern: ^[a-z0-9]([a-z0-9\.\-]*[a-z0-9])?$
- type: string
- passwordKey:
- default: password
- description: passwordKey the map key of the password in the
- connection credential secret
- type: string
- usernameKey:
- default: username
- description: usernameKey the map key of the user in the connection
- credential secret
- type: string
- required:
- - name
- type: object
- required:
- - labelsSelector
- type: object
- targetVolumeMounts:
- description: array of restore volume mounts .
- items:
- description: VolumeMount describes a mounting of a Volume within
- a container.
- properties:
- mountPath:
- description: Path within the container at which the volume should
- be mounted. Must not contain ':'.
- type: string
- mountPropagation:
- description: mountPropagation determines how mounts are propagated
- from the host to container and the other way around. When
- not set, MountPropagationNone is used. This field is beta
- in 1.10.
- type: string
- name:
- description: This must match the Name of a Volume.
- type: string
- readOnly:
- description: Mounted read-only if true, read-write otherwise
- (false or unspecified). Defaults to false.
- type: boolean
- subPath:
- description: Path within the volume from which the container's
- volume should be mounted. Defaults to "" (volume's root).
- type: string
- subPathExpr:
- description: Expanded path within the volume from which the
- container's volume should be mounted. Behaves similarly to
- SubPath but environment variable references $(VAR_NAME) are
- expanded using the container's environment. Defaults to ""
- (volume's root). SubPathExpr and SubPath are mutually exclusive.
- type: string
- required:
- - mountPath
- - name
- type: object
- minItems: 1
- type: array
- x-kubernetes-preserve-unknown-fields: true
- targetVolumes:
- description: array of restore volumes .
- items:
- description: Volume represents a named volume in a pod that may
- be accessed by any container in the pod.
- properties:
- awsElasticBlockStore:
- description: 'awsElasticBlockStore represents an AWS Disk resource
- that is attached to a kubelet''s host machine and then exposed
- to the pod. More info: https://kubernetes.io/docs/concepts/storage/volumes#awselasticblockstore'
- properties:
- fsType:
- description: 'fsType is the filesystem type of the volume
- that you want to mount. Tip: Ensure that the filesystem
- type is supported by the host operating system. Examples:
- "ext4", "xfs", "ntfs". Implicitly inferred to be "ext4"
- if unspecified. More info: https://kubernetes.io/docs/concepts/storage/volumes#awselasticblockstore
- TODO: how do we prevent errors in the filesystem from
- compromising the machine'
- type: string
- partition:
- description: 'partition is the partition in the volume that
- you want to mount. If omitted, the default is to mount
- by volume name. Examples: For volume /dev/sda1, you specify
- the partition as "1". Similarly, the volume partition
- for /dev/sda is "0" (or you can leave the property empty).'
- format: int32
- type: integer
- readOnly:
- description: 'readOnly value true will force the readOnly
- setting in VolumeMounts. More info: https://kubernetes.io/docs/concepts/storage/volumes#awselasticblockstore'
- type: boolean
- volumeID:
- description: 'volumeID is unique ID of the persistent disk
- resource in AWS (Amazon EBS volume). More info: https://kubernetes.io/docs/concepts/storage/volumes#awselasticblockstore'
- type: string
- required:
- - volumeID
- type: object
- azureDisk:
- description: azureDisk represents an Azure Data Disk mount on
- the host and bind mount to the pod.
- properties:
- cachingMode:
- description: 'cachingMode is the Host Caching mode: None,
- Read Only, Read Write.'
- type: string
- diskName:
- description: diskName is the Name of the data disk in the
- blob storage
- type: string
- diskURI:
- description: diskURI is the URI of data disk in the blob
- storage
- type: string
- fsType:
- description: fsType is Filesystem type to mount. Must be
- a filesystem type supported by the host operating system.
- Ex. "ext4", "xfs", "ntfs". Implicitly inferred to be "ext4"
- if unspecified.
- type: string
- kind:
- description: 'kind expected values are Shared: multiple
- blob disks per storage account Dedicated: single blob
- disk per storage account Managed: azure managed data
- disk (only in managed availability set). defaults to shared'
- type: string
- readOnly:
- description: readOnly Defaults to false (read/write). ReadOnly
- here will force the ReadOnly setting in VolumeMounts.
- type: boolean
- required:
- - diskName
- - diskURI
- type: object
- azureFile:
- description: azureFile represents an Azure File Service mount
- on the host and bind mount to the pod.
- properties:
- readOnly:
- description: readOnly defaults to false (read/write). ReadOnly
- here will force the ReadOnly setting in VolumeMounts.
- type: boolean
- secretName:
- description: secretName is the name of secret that contains
- Azure Storage Account Name and Key
- type: string
- shareName:
- description: shareName is the azure share Name
- type: string
- required:
- - secretName
- - shareName
- type: object
- cephfs:
- description: cephFS represents a Ceph FS mount on the host that
- shares a pod's lifetime
- properties:
- monitors:
- description: 'monitors is Required: Monitors is a collection
- of Ceph monitors More info: https://examples.k8s.io/volumes/cephfs/README.md#how-to-use-it'
- items:
- type: string
- type: array
- path:
- description: 'path is Optional: Used as the mounted root,
- rather than the full Ceph tree, default is /'
- type: string
- readOnly:
- description: 'readOnly is Optional: Defaults to false (read/write).
- ReadOnly here will force the ReadOnly setting in VolumeMounts.
- More info: https://examples.k8s.io/volumes/cephfs/README.md#how-to-use-it'
- type: boolean
- secretFile:
- description: 'secretFile is Optional: SecretFile is the
- path to key ring for User, default is /etc/ceph/user.secret
- More info: https://examples.k8s.io/volumes/cephfs/README.md#how-to-use-it'
- type: string
- secretRef:
- description: 'secretRef is Optional: SecretRef is reference
- to the authentication secret for User, default is empty.
- More info: https://examples.k8s.io/volumes/cephfs/README.md#how-to-use-it'
- properties:
- name:
- description: 'Name of the referent. More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names
- TODO: Add other useful fields. apiVersion, kind, uid?'
- type: string
- type: object
- user:
- description: 'user is optional: User is the rados user name,
- default is admin More info: https://examples.k8s.io/volumes/cephfs/README.md#how-to-use-it'
- type: string
- required:
- - monitors
- type: object
- cinder:
- description: 'cinder represents a cinder volume attached and
- mounted on kubelets host machine. More info: https://examples.k8s.io/mysql-cinder-pd/README.md'
- properties:
- fsType:
- description: 'fsType is the filesystem type to mount. Must
- be a filesystem type supported by the host operating system.
- Examples: "ext4", "xfs", "ntfs". Implicitly inferred to
- be "ext4" if unspecified. More info: https://examples.k8s.io/mysql-cinder-pd/README.md'
- type: string
- readOnly:
- description: 'readOnly defaults to false (read/write). ReadOnly
- here will force the ReadOnly setting in VolumeMounts.
- More info: https://examples.k8s.io/mysql-cinder-pd/README.md'
- type: boolean
- secretRef:
- description: 'secretRef is optional: points to a secret
- object containing parameters used to connect to OpenStack.'
- properties:
- name:
- description: 'Name of the referent. More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names
- TODO: Add other useful fields. apiVersion, kind, uid?'
- type: string
- type: object
- volumeID:
- description: 'volumeID used to identify the volume in cinder.
- More info: https://examples.k8s.io/mysql-cinder-pd/README.md'
- type: string
- required:
- - volumeID
- type: object
- configMap:
- description: configMap represents a configMap that should populate
- this volume
- properties:
- defaultMode:
- description: 'defaultMode is optional: mode bits used to
- set permissions on created files by default. Must be an
- octal value between 0000 and 0777 or a decimal value between
- 0 and 511. YAML accepts both octal and decimal values,
- JSON requires decimal values for mode bits. Defaults to
- 0644. Directories within the path are not affected by
- this setting. This might be in conflict with other options
- that affect the file mode, like fsGroup, and the result
- can be other mode bits set.'
- format: int32
- type: integer
- items:
- description: items if unspecified, each key-value pair in
- the Data field of the referenced ConfigMap will be projected
- into the volume as a file whose name is the key and content
- is the value. If specified, the listed keys will be projected
- into the specified paths, and unlisted keys will not be
- present. If a key is specified which is not present in
- the ConfigMap, the volume setup will error unless it is
- marked optional. Paths must be relative and may not contain
- the '..' path or start with '..'.
- items:
- description: Maps a string key to a path within a volume.
- properties:
- key:
- description: key is the key to project.
- type: string
- mode:
- description: 'mode is Optional: mode bits used to
- set permissions on this file. Must be an octal value
- between 0000 and 0777 or a decimal value between
- 0 and 511. YAML accepts both octal and decimal values,
- JSON requires decimal values for mode bits. If not
- specified, the volume defaultMode will be used.
- This might be in conflict with other options that
- affect the file mode, like fsGroup, and the result
- can be other mode bits set.'
- format: int32
- type: integer
- path:
- description: path is the relative path of the file
- to map the key to. May not be an absolute path.
- May not contain the path element '..'. May not start
- with the string '..'.
- type: string
- required:
- - key
- - path
- type: object
- type: array
- name:
- description: 'Name of the referent. More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names
- TODO: Add other useful fields. apiVersion, kind, uid?'
- type: string
- optional:
- description: optional specify whether the ConfigMap or its
- keys must be defined
- type: boolean
- type: object
- csi:
- description: csi (Container Storage Interface) represents ephemeral
- storage that is handled by certain external CSI drivers (Beta
- feature).
- properties:
- driver:
- description: driver is the name of the CSI driver that handles
- this volume. Consult with your admin for the correct name
- as registered in the cluster.
- type: string
- fsType:
- description: fsType to mount. Ex. "ext4", "xfs", "ntfs".
- If not provided, the empty value is passed to the associated
- CSI driver which will determine the default filesystem
- to apply.
- type: string
- nodePublishSecretRef:
- description: nodePublishSecretRef is a reference to the
- secret object containing sensitive information to pass
- to the CSI driver to complete the CSI NodePublishVolume
- and NodeUnpublishVolume calls. This field is optional,
- and may be empty if no secret is required. If the secret
- object contains more than one secret, all secret references
- are passed.
- properties:
- name:
- description: 'Name of the referent. More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names
- TODO: Add other useful fields. apiVersion, kind, uid?'
- type: string
- type: object
- readOnly:
- description: readOnly specifies a read-only configuration
- for the volume. Defaults to false (read/write).
- type: boolean
- volumeAttributes:
- additionalProperties:
- type: string
- description: volumeAttributes stores driver-specific properties
- that are passed to the CSI driver. Consult your driver's
- documentation for supported values.
- type: object
- required:
- - driver
- type: object
- downwardAPI:
- description: downwardAPI represents downward API about the pod
- that should populate this volume
- properties:
- defaultMode:
- description: 'Optional: mode bits to use on created files
- by default. Must be a Optional: mode bits used to set
- permissions on created files by default. Must be an octal
- value between 0000 and 0777 or a decimal value between
- 0 and 511. YAML accepts both octal and decimal values,
- JSON requires decimal values for mode bits. Defaults to
- 0644. Directories within the path are not affected by
- this setting. This might be in conflict with other options
- that affect the file mode, like fsGroup, and the result
- can be other mode bits set.'
- format: int32
- type: integer
- items:
- description: Items is a list of downward API volume file
- items:
- description: DownwardAPIVolumeFile represents information
- to create the file containing the pod field
- properties:
- fieldRef:
- description: 'Required: Selects a field of the pod:
- only annotations, labels, name and namespace are
- supported.'
- properties:
- apiVersion:
- description: Version of the schema the FieldPath
- is written in terms of, defaults to "v1".
- type: string
- fieldPath:
- description: Path of the field to select in the
- specified API version.
- type: string
- required:
- - fieldPath
- type: object
- mode:
- description: 'Optional: mode bits used to set permissions
- on this file, must be an octal value between 0000
- and 0777 or a decimal value between 0 and 511. YAML
- accepts both octal and decimal values, JSON requires
- decimal values for mode bits. If not specified,
- the volume defaultMode will be used. This might
- be in conflict with other options that affect the
- file mode, like fsGroup, and the result can be other
- mode bits set.'
- format: int32
- type: integer
- path:
- description: 'Required: Path is the relative path
- name of the file to be created. Must not be absolute
- or contain the ''..'' path. Must be utf-8 encoded.
- The first item of the relative path must not start
- with ''..'''
- type: string
- resourceFieldRef:
- description: 'Selects a resource of the container:
- only resources limits and requests (limits.cpu,
- limits.memory, requests.cpu and requests.memory)
- are currently supported.'
- properties:
- containerName:
- description: 'Container name: required for volumes,
- optional for env vars'
- type: string
- divisor:
- anyOf:
- - type: integer
- - type: string
- description: Specifies the output format of the
- exposed resources, defaults to "1"
- pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$
- x-kubernetes-int-or-string: true
- resource:
- description: 'Required: resource to select'
- type: string
- required:
- - resource
- type: object
- required:
- - path
- type: object
- type: array
- type: object
- emptyDir:
- description: 'emptyDir represents a temporary directory that
- shares a pod''s lifetime. More info: https://kubernetes.io/docs/concepts/storage/volumes#emptydir'
- properties:
- medium:
- description: 'medium represents what type of storage medium
- should back this directory. The default is "" which means
- to use the node''s default medium. Must be an empty string
- (default) or Memory. More info: https://kubernetes.io/docs/concepts/storage/volumes#emptydir'
- type: string
- sizeLimit:
- anyOf:
- - type: integer
- - type: string
- description: 'sizeLimit is the total amount of local storage
- required for this EmptyDir volume. The size limit is also
- applicable for memory medium. The maximum usage on memory
- medium EmptyDir would be the minimum value between the
- SizeLimit specified here and the sum of memory limits
- of all containers in a pod. The default is nil which means
- that the limit is undefined. More info: http://kubernetes.io/docs/user-guide/volumes#emptydir'
- pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$
- x-kubernetes-int-or-string: true
- type: object
- ephemeral:
- description: "ephemeral represents a volume that is handled
- by a cluster storage driver. The volume's lifecycle is tied
- to the pod that defines it - it will be created before the
- pod starts, and deleted when the pod is removed. \n Use this
- if: a) the volume is only needed while the pod runs, b) features
- of normal volumes like restoring from snapshot or capacity
- tracking are needed, c) the storage driver is specified through
- a storage class, and d) the storage driver supports dynamic
- volume provisioning through a PersistentVolumeClaim (see EphemeralVolumeSource
- for more information on the connection between this volume
- type and PersistentVolumeClaim). \n Use PersistentVolumeClaim
- or one of the vendor-specific APIs for volumes that persist
- for longer than the lifecycle of an individual pod. \n Use
- CSI for light-weight local ephemeral volumes if the CSI driver
- is meant to be used that way - see the documentation of the
- driver for more information. \n A pod can use both types of
- ephemeral volumes and persistent volumes at the same time."
- properties:
- volumeClaimTemplate:
- description: "Will be used to create a stand-alone PVC to
- provision the volume. The pod in which this EphemeralVolumeSource
- is embedded will be the owner of the PVC, i.e. the PVC
- will be deleted together with the pod. The name of the
- PVC will be `-` where `` is the name from the `PodSpec.Volumes` array entry.
- Pod validation will reject the pod if the concatenated
- name is not valid for a PVC (for example, too long). \n
- An existing PVC with that name that is not owned by the
- pod will *not* be used for the pod to avoid using an unrelated
- volume by mistake. Starting the pod is then blocked until
- the unrelated PVC is removed. If such a pre-created PVC
- is meant to be used by the pod, the PVC has to updated
- with an owner reference to the pod once the pod exists.
- Normally this should not be necessary, but it may be useful
- when manually reconstructing a broken cluster. \n This
- field is read-only and no changes will be made by Kubernetes
- to the PVC after it has been created. \n Required, must
- not be nil."
- properties:
- metadata:
- description: May contain labels and annotations that
- will be copied into the PVC when creating it. No other
- fields are allowed and will be rejected during validation.
- properties:
- annotations:
- additionalProperties:
- type: string
- type: object
- finalizers:
- items:
- type: string
- type: array
- labels:
- additionalProperties:
- type: string
- type: object
- name:
- type: string
- namespace:
- type: string
- type: object
- spec:
- description: The specification for the PersistentVolumeClaim.
- The entire content is copied unchanged into the PVC
- that gets created from this template. The same fields
- as in a PersistentVolumeClaim are also valid here.
- properties:
- accessModes:
- description: 'accessModes contains the desired access
- modes the volume should have. More info: https://kubernetes.io/docs/concepts/storage/persistent-volumes#access-modes-1'
- items:
- type: string
- type: array
- dataSource:
- description: 'dataSource field can be used to specify
- either: * An existing VolumeSnapshot object (snapshot.storage.k8s.io/VolumeSnapshot)
- * An existing PVC (PersistentVolumeClaim) If the
- provisioner or an external controller can support
- the specified data source, it will create a new
- volume based on the contents of the specified
- data source. When the AnyVolumeDataSource feature
- gate is enabled, dataSource contents will be copied
- to dataSourceRef, and dataSourceRef contents will
- be copied to dataSource when dataSourceRef.namespace
- is not specified. If the namespace is specified,
- then dataSourceRef will not be copied to dataSource.'
- properties:
- apiGroup:
- description: APIGroup is the group for the resource
- being referenced. If APIGroup is not specified,
- the specified Kind must be in the core API
- group. For any other third-party types, APIGroup
- is required.
- type: string
- kind:
- description: Kind is the type of resource being
- referenced
- type: string
- name:
- description: Name is the name of resource being
- referenced
- type: string
- required:
- - kind
- - name
- type: object
- dataSourceRef:
- description: 'dataSourceRef specifies the object
- from which to populate the volume with data, if
- a non-empty volume is desired. This may be any
- object from a non-empty API group (non core object)
- or a PersistentVolumeClaim object. When this field
- is specified, volume binding will only succeed
- if the type of the specified object matches some
- installed volume populator or dynamic provisioner.
- This field will replace the functionality of the
- dataSource field and as such if both fields are
- non-empty, they must have the same value. For
- backwards compatibility, when namespace isn''t
- specified in dataSourceRef, both fields (dataSource
- and dataSourceRef) will be set to the same value
- automatically if one of them is empty and the
- other is non-empty. When namespace is specified
- in dataSourceRef, dataSource isn''t set to the
- same value and must be empty. There are three
- important differences between dataSource and dataSourceRef:
- * While dataSource only allows two specific types
- of objects, dataSourceRef allows any non-core
- object, as well as PersistentVolumeClaim objects.
- * While dataSource ignores disallowed values (dropping
- them), dataSourceRef preserves all values, and
- generates an error if a disallowed value is specified.
- * While dataSource only allows local objects,
- dataSourceRef allows objects in any namespaces.
- (Beta) Using this field requires the AnyVolumeDataSource
- feature gate to be enabled. (Alpha) Using the
- namespace field of dataSourceRef requires the
- CrossNamespaceVolumeDataSource feature gate to
- be enabled.'
- properties:
- apiGroup:
- description: APIGroup is the group for the resource
- being referenced. If APIGroup is not specified,
- the specified Kind must be in the core API
- group. For any other third-party types, APIGroup
- is required.
- type: string
- kind:
- description: Kind is the type of resource being
- referenced
- type: string
- name:
- description: Name is the name of resource being
- referenced
- type: string
- namespace:
- description: Namespace is the namespace of resource
- being referenced Note that when a namespace
- is specified, a gateway.networking.k8s.io/ReferenceGrant
- object is required in the referent namespace
- to allow that namespace's owner to accept
- the reference. See the ReferenceGrant documentation
- for details. (Alpha) This field requires the
- CrossNamespaceVolumeDataSource feature gate
- to be enabled.
- type: string
- required:
- - kind
- - name
- type: object
- resources:
- description: 'resources represents the minimum resources
- the volume should have. If RecoverVolumeExpansionFailure
- feature is enabled users are allowed to specify
- resource requirements that are lower than previous
- value but must still be higher than capacity recorded
- in the status field of the claim. More info: https://kubernetes.io/docs/concepts/storage/persistent-volumes#resources'
- properties:
- claims:
- description: "Claims lists the names of resources,
- defined in spec.resourceClaims, that are used
- by this container. \n This is an alpha field
- and requires enabling the DynamicResourceAllocation
- feature gate. \n This field is immutable.
- It can only be set for containers."
- items:
- description: ResourceClaim references one
- entry in PodSpec.ResourceClaims.
- properties:
- name:
- description: Name must match the name
- of one entry in pod.spec.resourceClaims
- of the Pod where this field is used.
- It makes that resource available inside
- a container.
- type: string
- required:
- - name
- type: object
- type: array
- x-kubernetes-list-map-keys:
- - name
- x-kubernetes-list-type: map
- limits:
- additionalProperties:
- anyOf:
- - type: integer
- - type: string
- pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$
- x-kubernetes-int-or-string: true
- description: 'Limits describes the maximum amount
- of compute resources allowed. More info: https://kubernetes.io/docs/concepts/configuration/manage-resources-containers/'
- type: object
- requests:
- additionalProperties:
- anyOf:
- - type: integer
- - type: string
- pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$
- x-kubernetes-int-or-string: true
- description: 'Requests describes the minimum
- amount of compute resources required. If Requests
- is omitted for a container, it defaults to
- Limits if that is explicitly specified, otherwise
- to an implementation-defined value. More info:
- https://kubernetes.io/docs/concepts/configuration/manage-resources-containers/'
- type: object
- type: object
- selector:
- description: selector is a label query over volumes
- to consider for binding.
- properties:
- matchExpressions:
- description: matchExpressions is a list of label
- selector requirements. The requirements are
- ANDed.
- items:
- description: A label selector requirement
- is a selector that contains values, a key,
- and an operator that relates the key and
- values.
- properties:
- key:
- description: key is the label key that
- the selector applies to.
- type: string
- operator:
- description: operator represents a key's
- relationship to a set of values. Valid
- operators are In, NotIn, Exists and
- DoesNotExist.
- type: string
- values:
- description: values is an array of string
- values. If the operator is In or NotIn,
- the values array must be non-empty.
- If the operator is Exists or DoesNotExist,
- the values array must be empty. This
- array is replaced during a strategic
- merge patch.
- items:
- type: string
- type: array
- required:
- - key
- - operator
- type: object
- type: array
- matchLabels:
- additionalProperties:
- type: string
- description: matchLabels is a map of {key,value}
- pairs. A single {key,value} in the matchLabels
- map is equivalent to an element of matchExpressions,
- whose key field is "key", the operator is
- "In", and the values array contains only "value".
- The requirements are ANDed.
- type: object
- type: object
- storageClassName:
- description: 'storageClassName is the name of the
- StorageClass required by the claim. More info:
- https://kubernetes.io/docs/concepts/storage/persistent-volumes#class-1'
- type: string
- volumeMode:
- description: volumeMode defines what type of volume
- is required by the claim. Value of Filesystem
- is implied when not included in claim spec.
- type: string
- volumeName:
- description: volumeName is the binding reference
- to the PersistentVolume backing this claim.
- type: string
- type: object
- required:
- - spec
- type: object
- type: object
- fc:
- description: fc represents a Fibre Channel resource that is
- attached to a kubelet's host machine and then exposed to the
- pod.
- properties:
- fsType:
- description: 'fsType is the filesystem type to mount. Must
- be a filesystem type supported by the host operating system.
- Ex. "ext4", "xfs", "ntfs". Implicitly inferred to be "ext4"
- if unspecified. TODO: how do we prevent errors in the
- filesystem from compromising the machine'
- type: string
- lun:
- description: 'lun is Optional: FC target lun number'
- format: int32
- type: integer
- readOnly:
- description: 'readOnly is Optional: Defaults to false (read/write).
- ReadOnly here will force the ReadOnly setting in VolumeMounts.'
- type: boolean
- targetWWNs:
- description: 'targetWWNs is Optional: FC target worldwide
- names (WWNs)'
- items:
- type: string
- type: array
- wwids:
- description: 'wwids Optional: FC volume world wide identifiers
- (wwids) Either wwids or combination of targetWWNs and
- lun must be set, but not both simultaneously.'
- items:
- type: string
- type: array
- type: object
- flexVolume:
- description: flexVolume represents a generic volume resource
- that is provisioned/attached using an exec based plugin.
- properties:
- driver:
- description: driver is the name of the driver to use for
- this volume.
- type: string
- fsType:
- description: fsType is the filesystem type to mount. Must
- be a filesystem type supported by the host operating system.
- Ex. "ext4", "xfs", "ntfs". The default filesystem depends
- on FlexVolume script.
- type: string
- options:
- additionalProperties:
- type: string
- description: 'options is Optional: this field holds extra
- command options if any.'
- type: object
- readOnly:
- description: 'readOnly is Optional: defaults to false (read/write).
- ReadOnly here will force the ReadOnly setting in VolumeMounts.'
- type: boolean
- secretRef:
- description: 'secretRef is Optional: secretRef is reference
- to the secret object containing sensitive information
- to pass to the plugin scripts. This may be empty if no
- secret object is specified. If the secret object contains
- more than one secret, all secrets are passed to the plugin
- scripts.'
- properties:
- name:
- description: 'Name of the referent. More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names
- TODO: Add other useful fields. apiVersion, kind, uid?'
- type: string
- type: object
- required:
- - driver
- type: object
- flocker:
- description: flocker represents a Flocker volume attached to
- a kubelet's host machine. This depends on the Flocker control
- service being running
- properties:
- datasetName:
- description: datasetName is Name of the dataset stored as
- metadata -> name on the dataset for Flocker should be
- considered as deprecated
- type: string
- datasetUUID:
- description: datasetUUID is the UUID of the dataset. This
- is unique identifier of a Flocker dataset
- type: string
- type: object
- gcePersistentDisk:
- description: 'gcePersistentDisk represents a GCE Disk resource
- that is attached to a kubelet''s host machine and then exposed
- to the pod. More info: https://kubernetes.io/docs/concepts/storage/volumes#gcepersistentdisk'
- properties:
- fsType:
- description: 'fsType is filesystem type of the volume that
- you want to mount. Tip: Ensure that the filesystem type
- is supported by the host operating system. Examples: "ext4",
- "xfs", "ntfs". Implicitly inferred to be "ext4" if unspecified.
- More info: https://kubernetes.io/docs/concepts/storage/volumes#gcepersistentdisk
- TODO: how do we prevent errors in the filesystem from
- compromising the machine'
- type: string
- partition:
- description: 'partition is the partition in the volume that
- you want to mount. If omitted, the default is to mount
- by volume name. Examples: For volume /dev/sda1, you specify
- the partition as "1". Similarly, the volume partition
- for /dev/sda is "0" (or you can leave the property empty).
- More info: https://kubernetes.io/docs/concepts/storage/volumes#gcepersistentdisk'
- format: int32
- type: integer
- pdName:
- description: 'pdName is unique name of the PD resource in
- GCE. Used to identify the disk in GCE. More info: https://kubernetes.io/docs/concepts/storage/volumes#gcepersistentdisk'
- type: string
- readOnly:
- description: 'readOnly here will force the ReadOnly setting
- in VolumeMounts. Defaults to false. More info: https://kubernetes.io/docs/concepts/storage/volumes#gcepersistentdisk'
- type: boolean
- required:
- - pdName
- type: object
- gitRepo:
- description: 'gitRepo represents a git repository at a particular
- revision. DEPRECATED: GitRepo is deprecated. To provision
- a container with a git repo, mount an EmptyDir into an InitContainer
- that clones the repo using git, then mount the EmptyDir into
- the Pod''s container.'
- properties:
- directory:
- description: directory is the target directory name. Must
- not contain or start with '..'. If '.' is supplied, the
- volume directory will be the git repository. Otherwise,
- if specified, the volume will contain the git repository
- in the subdirectory with the given name.
- type: string
- repository:
- description: repository is the URL
- type: string
- revision:
- description: revision is the commit hash for the specified
- revision.
- type: string
- required:
- - repository
- type: object
- glusterfs:
- description: 'glusterfs represents a Glusterfs mount on the
- host that shares a pod''s lifetime. More info: https://examples.k8s.io/volumes/glusterfs/README.md'
- properties:
- endpoints:
- description: 'endpoints is the endpoint name that details
- Glusterfs topology. More info: https://examples.k8s.io/volumes/glusterfs/README.md#create-a-pod'
- type: string
- path:
- description: 'path is the Glusterfs volume path. More info:
- https://examples.k8s.io/volumes/glusterfs/README.md#create-a-pod'
- type: string
- readOnly:
- description: 'readOnly here will force the Glusterfs volume
- to be mounted with read-only permissions. Defaults to
- false. More info: https://examples.k8s.io/volumes/glusterfs/README.md#create-a-pod'
- type: boolean
- required:
- - endpoints
- - path
- type: object
- hostPath:
- description: 'hostPath represents a pre-existing file or directory
- on the host machine that is directly exposed to the container.
- This is generally used for system agents or other privileged
- things that are allowed to see the host machine. Most containers
- will NOT need this. More info: https://kubernetes.io/docs/concepts/storage/volumes#hostpath
- --- TODO(jonesdl) We need to restrict who can use host directory
- mounts and who can/can not mount host directories as read/write.'
- properties:
- path:
- description: 'path of the directory on the host. If the
- path is a symlink, it will follow the link to the real
- path. More info: https://kubernetes.io/docs/concepts/storage/volumes#hostpath'
- type: string
- type:
- description: 'type for HostPath Volume Defaults to "" More
- info: https://kubernetes.io/docs/concepts/storage/volumes#hostpath'
- type: string
- required:
- - path
- type: object
- iscsi:
- description: 'iscsi represents an ISCSI Disk resource that is
- attached to a kubelet''s host machine and then exposed to
- the pod. More info: https://examples.k8s.io/volumes/iscsi/README.md'
- properties:
- chapAuthDiscovery:
- description: chapAuthDiscovery defines whether support iSCSI
- Discovery CHAP authentication
- type: boolean
- chapAuthSession:
- description: chapAuthSession defines whether support iSCSI
- Session CHAP authentication
- type: boolean
- fsType:
- description: 'fsType is the filesystem type of the volume
- that you want to mount. Tip: Ensure that the filesystem
- type is supported by the host operating system. Examples:
- "ext4", "xfs", "ntfs". Implicitly inferred to be "ext4"
- if unspecified. More info: https://kubernetes.io/docs/concepts/storage/volumes#iscsi
- TODO: how do we prevent errors in the filesystem from
- compromising the machine'
- type: string
- initiatorName:
- description: initiatorName is the custom iSCSI Initiator
- Name. If initiatorName is specified with iscsiInterface
- simultaneously, new iSCSI interface : will be created for the connection.
- type: string
- iqn:
- description: iqn is the target iSCSI Qualified Name.
- type: string
- iscsiInterface:
- description: iscsiInterface is the interface Name that uses
- an iSCSI transport. Defaults to 'default' (tcp).
- type: string
- lun:
- description: lun represents iSCSI Target Lun number.
- format: int32
- type: integer
- portals:
- description: portals is the iSCSI Target Portal List. The
- portal is either an IP or ip_addr:port if the port is
- other than default (typically TCP ports 860 and 3260).
- items:
- type: string
- type: array
- readOnly:
- description: readOnly here will force the ReadOnly setting
- in VolumeMounts. Defaults to false.
- type: boolean
- secretRef:
- description: secretRef is the CHAP Secret for iSCSI target
- and initiator authentication
- properties:
- name:
- description: 'Name of the referent. More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names
- TODO: Add other useful fields. apiVersion, kind, uid?'
- type: string
- type: object
- targetPortal:
- description: targetPortal is iSCSI Target Portal. The Portal
- is either an IP or ip_addr:port if the port is other than
- default (typically TCP ports 860 and 3260).
- type: string
- required:
- - iqn
- - lun
- - targetPortal
- type: object
- name:
- description: 'name of the volume. Must be a DNS_LABEL and unique
- within the pod. More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names'
- type: string
- nfs:
- description: 'nfs represents an NFS mount on the host that shares
- a pod''s lifetime More info: https://kubernetes.io/docs/concepts/storage/volumes#nfs'
- properties:
- path:
- description: 'path that is exported by the NFS server. More
- info: https://kubernetes.io/docs/concepts/storage/volumes#nfs'
- type: string
- readOnly:
- description: 'readOnly here will force the NFS export to
- be mounted with read-only permissions. Defaults to false.
- More info: https://kubernetes.io/docs/concepts/storage/volumes#nfs'
- type: boolean
- server:
- description: 'server is the hostname or IP address of the
- NFS server. More info: https://kubernetes.io/docs/concepts/storage/volumes#nfs'
- type: string
- required:
- - path
- - server
- type: object
- persistentVolumeClaim:
- description: 'persistentVolumeClaimVolumeSource represents a
- reference to a PersistentVolumeClaim in the same namespace.
- More info: https://kubernetes.io/docs/concepts/storage/persistent-volumes#persistentvolumeclaims'
- properties:
- claimName:
- description: 'claimName is the name of a PersistentVolumeClaim
- in the same namespace as the pod using this volume. More
- info: https://kubernetes.io/docs/concepts/storage/persistent-volumes#persistentvolumeclaims'
- type: string
- readOnly:
- description: readOnly Will force the ReadOnly setting in
- VolumeMounts. Default false.
- type: boolean
- required:
- - claimName
- type: object
- photonPersistentDisk:
- description: photonPersistentDisk represents a PhotonController
- persistent disk attached and mounted on kubelets host machine
- properties:
- fsType:
- description: fsType is the filesystem type to mount. Must
- be a filesystem type supported by the host operating system.
- Ex. "ext4", "xfs", "ntfs". Implicitly inferred to be "ext4"
- if unspecified.
- type: string
- pdID:
- description: pdID is the ID that identifies Photon Controller
- persistent disk
- type: string
- required:
- - pdID
- type: object
- portworxVolume:
- description: portworxVolume represents a portworx volume attached
- and mounted on kubelets host machine
- properties:
- fsType:
- description: fSType represents the filesystem type to mount
- Must be a filesystem type supported by the host operating
- system. Ex. "ext4", "xfs". Implicitly inferred to be "ext4"
- if unspecified.
- type: string
- readOnly:
- description: readOnly defaults to false (read/write). ReadOnly
- here will force the ReadOnly setting in VolumeMounts.
- type: boolean
- volumeID:
- description: volumeID uniquely identifies a Portworx volume
- type: string
- required:
- - volumeID
- type: object
- projected:
- description: projected items for all in one resources secrets,
- configmaps, and downward API
- properties:
- defaultMode:
- description: defaultMode are the mode bits used to set permissions
- on created files by default. Must be an octal value between
- 0000 and 0777 or a decimal value between 0 and 511. YAML
- accepts both octal and decimal values, JSON requires decimal
- values for mode bits. Directories within the path are
- not affected by this setting. This might be in conflict
- with other options that affect the file mode, like fsGroup,
- and the result can be other mode bits set.
- format: int32
- type: integer
- sources:
- description: sources is the list of volume projections
- items:
- description: Projection that may be projected along with
- other supported volume types
- properties:
- configMap:
- description: configMap information about the configMap
- data to project
- properties:
- items:
- description: items if unspecified, each key-value
- pair in the Data field of the referenced ConfigMap
- will be projected into the volume as a file
- whose name is the key and content is the value.
- If specified, the listed keys will be projected
- into the specified paths, and unlisted keys
- will not be present. If a key is specified which
- is not present in the ConfigMap, the volume
- setup will error unless it is marked optional.
- Paths must be relative and may not contain the
- '..' path or start with '..'.
- items:
- description: Maps a string key to a path within
- a volume.
- properties:
- key:
- description: key is the key to project.
- type: string
- mode:
- description: 'mode is Optional: mode bits
- used to set permissions on this file.
- Must be an octal value between 0000 and
- 0777 or a decimal value between 0 and
- 511. YAML accepts both octal and decimal
- values, JSON requires decimal values for
- mode bits. If not specified, the volume
- defaultMode will be used. This might be
- in conflict with other options that affect
- the file mode, like fsGroup, and the result
- can be other mode bits set.'
- format: int32
- type: integer
- path:
- description: path is the relative path of
- the file to map the key to. May not be
- an absolute path. May not contain the
- path element '..'. May not start with
- the string '..'.
- type: string
- required:
- - key
- - path
- type: object
- type: array
- name:
- description: 'Name of the referent. More info:
- https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names
- TODO: Add other useful fields. apiVersion, kind,
- uid?'
- type: string
- optional:
- description: optional specify whether the ConfigMap
- or its keys must be defined
- type: boolean
- type: object
- downwardAPI:
- description: downwardAPI information about the downwardAPI
- data to project
- properties:
- items:
- description: Items is a list of DownwardAPIVolume
- file
- items:
- description: DownwardAPIVolumeFile represents
- information to create the file containing
- the pod field
- properties:
- fieldRef:
- description: 'Required: Selects a field
- of the pod: only annotations, labels,
- name and namespace are supported.'
- properties:
- apiVersion:
- description: Version of the schema the
- FieldPath is written in terms of,
- defaults to "v1".
- type: string
- fieldPath:
- description: Path of the field to select
- in the specified API version.
- type: string
- required:
- - fieldPath
- type: object
- mode:
- description: 'Optional: mode bits used to
- set permissions on this file, must be
- an octal value between 0000 and 0777 or
- a decimal value between 0 and 511. YAML
- accepts both octal and decimal values,
- JSON requires decimal values for mode
- bits. If not specified, the volume defaultMode
- will be used. This might be in conflict
- with other options that affect the file
- mode, like fsGroup, and the result can
- be other mode bits set.'
- format: int32
- type: integer
- path:
- description: 'Required: Path is the relative
- path name of the file to be created. Must
- not be absolute or contain the ''..''
- path. Must be utf-8 encoded. The first
- item of the relative path must not start
- with ''..'''
- type: string
- resourceFieldRef:
- description: 'Selects a resource of the
- container: only resources limits and requests
- (limits.cpu, limits.memory, requests.cpu
- and requests.memory) are currently supported.'
- properties:
- containerName:
- description: 'Container name: required
- for volumes, optional for env vars'
- type: string
- divisor:
- anyOf:
- - type: integer
- - type: string
- description: Specifies the output format
- of the exposed resources, defaults
- to "1"
- pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$
- x-kubernetes-int-or-string: true
- resource:
- description: 'Required: resource to
- select'
- type: string
- required:
- - resource
- type: object
- required:
- - path
- type: object
- type: array
- type: object
- secret:
- description: secret information about the secret data
- to project
- properties:
- items:
- description: items if unspecified, each key-value
- pair in the Data field of the referenced Secret
- will be projected into the volume as a file
- whose name is the key and content is the value.
- If specified, the listed keys will be projected
- into the specified paths, and unlisted keys
- will not be present. If a key is specified which
- is not present in the Secret, the volume setup
- will error unless it is marked optional. Paths
- must be relative and may not contain the '..'
- path or start with '..'.
- items:
- description: Maps a string key to a path within
- a volume.
- properties:
- key:
- description: key is the key to project.
- type: string
- mode:
- description: 'mode is Optional: mode bits
- used to set permissions on this file.
- Must be an octal value between 0000 and
- 0777 or a decimal value between 0 and
- 511. YAML accepts both octal and decimal
- values, JSON requires decimal values for
- mode bits. If not specified, the volume
- defaultMode will be used. This might be
- in conflict with other options that affect
- the file mode, like fsGroup, and the result
- can be other mode bits set.'
- format: int32
- type: integer
- path:
- description: path is the relative path of
- the file to map the key to. May not be
- an absolute path. May not contain the
- path element '..'. May not start with
- the string '..'.
- type: string
- required:
- - key
- - path
- type: object
- type: array
- name:
- description: 'Name of the referent. More info:
- https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names
- TODO: Add other useful fields. apiVersion, kind,
- uid?'
- type: string
- optional:
- description: optional field specify whether the
- Secret or its key must be defined
- type: boolean
- type: object
- serviceAccountToken:
- description: serviceAccountToken is information about
- the serviceAccountToken data to project
- properties:
- audience:
- description: audience is the intended audience
- of the token. A recipient of a token must identify
- itself with an identifier specified in the audience
- of the token, and otherwise should reject the
- token. The audience defaults to the identifier
- of the apiserver.
- type: string
- expirationSeconds:
- description: expirationSeconds is the requested
- duration of validity of the service account
- token. As the token approaches expiration, the
- kubelet volume plugin will proactively rotate
- the service account token. The kubelet will
- start trying to rotate the token if the token
- is older than 80 percent of its time to live
- or if the token is older than 24 hours.Defaults
- to 1 hour and must be at least 10 minutes.
- format: int64
- type: integer
- path:
- description: path is the path relative to the
- mount point of the file to project the token
- into.
- type: string
- required:
- - path
- type: object
- type: object
- type: array
- type: object
- quobyte:
- description: quobyte represents a Quobyte mount on the host
- that shares a pod's lifetime
- properties:
- group:
- description: group to map volume access to Default is no
- group
- type: string
- readOnly:
- description: readOnly here will force the Quobyte volume
- to be mounted with read-only permissions. Defaults to
- false.
- type: boolean
- registry:
- description: registry represents a single or multiple Quobyte
- Registry services specified as a string as host:port pair
- (multiple entries are separated with commas) which acts
- as the central registry for volumes
- type: string
- tenant:
- description: tenant owning the given Quobyte volume in the
- Backend Used with dynamically provisioned Quobyte volumes,
- value is set by the plugin
- type: string
- user:
- description: user to map volume access to Defaults to serivceaccount
- user
- type: string
- volume:
- description: volume is a string that references an already
- created Quobyte volume by name.
- type: string
- required:
- - registry
- - volume
- type: object
- rbd:
- description: 'rbd represents a Rados Block Device mount on the
- host that shares a pod''s lifetime. More info: https://examples.k8s.io/volumes/rbd/README.md'
- properties:
- fsType:
- description: 'fsType is the filesystem type of the volume
- that you want to mount. Tip: Ensure that the filesystem
- type is supported by the host operating system. Examples:
- "ext4", "xfs", "ntfs". Implicitly inferred to be "ext4"
- if unspecified. More info: https://kubernetes.io/docs/concepts/storage/volumes#rbd
- TODO: how do we prevent errors in the filesystem from
- compromising the machine'
- type: string
- image:
- description: 'image is the rados image name. More info:
- https://examples.k8s.io/volumes/rbd/README.md#how-to-use-it'
- type: string
- keyring:
- description: 'keyring is the path to key ring for RBDUser.
- Default is /etc/ceph/keyring. More info: https://examples.k8s.io/volumes/rbd/README.md#how-to-use-it'
- type: string
- monitors:
- description: 'monitors is a collection of Ceph monitors.
- More info: https://examples.k8s.io/volumes/rbd/README.md#how-to-use-it'
- items:
- type: string
- type: array
- pool:
- description: 'pool is the rados pool name. Default is rbd.
- More info: https://examples.k8s.io/volumes/rbd/README.md#how-to-use-it'
- type: string
- readOnly:
- description: 'readOnly here will force the ReadOnly setting
- in VolumeMounts. Defaults to false. More info: https://examples.k8s.io/volumes/rbd/README.md#how-to-use-it'
- type: boolean
- secretRef:
- description: 'secretRef is name of the authentication secret
- for RBDUser. If provided overrides keyring. Default is
- nil. More info: https://examples.k8s.io/volumes/rbd/README.md#how-to-use-it'
- properties:
- name:
- description: 'Name of the referent. More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names
- TODO: Add other useful fields. apiVersion, kind, uid?'
- type: string
- type: object
- user:
- description: 'user is the rados user name. Default is admin.
- More info: https://examples.k8s.io/volumes/rbd/README.md#how-to-use-it'
- type: string
- required:
- - image
- - monitors
- type: object
- scaleIO:
- description: scaleIO represents a ScaleIO persistent volume
- attached and mounted on Kubernetes nodes.
- properties:
- fsType:
- description: fsType is the filesystem type to mount. Must
- be a filesystem type supported by the host operating system.
- Ex. "ext4", "xfs", "ntfs". Default is "xfs".
- type: string
- gateway:
- description: gateway is the host address of the ScaleIO
- API Gateway.
- type: string
- protectionDomain:
- description: protectionDomain is the name of the ScaleIO
- Protection Domain for the configured storage.
- type: string
- readOnly:
- description: readOnly Defaults to false (read/write). ReadOnly
- here will force the ReadOnly setting in VolumeMounts.
- type: boolean
- secretRef:
- description: secretRef references to the secret for ScaleIO
- user and other sensitive information. If this is not provided,
- Login operation will fail.
- properties:
- name:
- description: 'Name of the referent. More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names
- TODO: Add other useful fields. apiVersion, kind, uid?'
- type: string
- type: object
- sslEnabled:
- description: sslEnabled Flag enable/disable SSL communication
- with Gateway, default false
- type: boolean
- storageMode:
- description: storageMode indicates whether the storage for
- a volume should be ThickProvisioned or ThinProvisioned.
- Default is ThinProvisioned.
- type: string
- storagePool:
- description: storagePool is the ScaleIO Storage Pool associated
- with the protection domain.
- type: string
- system:
- description: system is the name of the storage system as
- configured in ScaleIO.
- type: string
- volumeName:
- description: volumeName is the name of a volume already
- created in the ScaleIO system that is associated with
- this volume source.
- type: string
- required:
- - gateway
- - secretRef
- - system
- type: object
- secret:
- description: 'secret represents a secret that should populate
- this volume. More info: https://kubernetes.io/docs/concepts/storage/volumes#secret'
- properties:
- defaultMode:
- description: 'defaultMode is Optional: mode bits used to
- set permissions on created files by default. Must be an
- octal value between 0000 and 0777 or a decimal value between
- 0 and 511. YAML accepts both octal and decimal values,
- JSON requires decimal values for mode bits. Defaults to
- 0644. Directories within the path are not affected by
- this setting. This might be in conflict with other options
- that affect the file mode, like fsGroup, and the result
- can be other mode bits set.'
- format: int32
- type: integer
- items:
- description: items If unspecified, each key-value pair in
- the Data field of the referenced Secret will be projected
- into the volume as a file whose name is the key and content
- is the value. If specified, the listed keys will be projected
- into the specified paths, and unlisted keys will not be
- present. If a key is specified which is not present in
- the Secret, the volume setup will error unless it is marked
- optional. Paths must be relative and may not contain the
- '..' path or start with '..'.
- items:
- description: Maps a string key to a path within a volume.
- properties:
- key:
- description: key is the key to project.
- type: string
- mode:
- description: 'mode is Optional: mode bits used to
- set permissions on this file. Must be an octal value
- between 0000 and 0777 or a decimal value between
- 0 and 511. YAML accepts both octal and decimal values,
- JSON requires decimal values for mode bits. If not
- specified, the volume defaultMode will be used.
- This might be in conflict with other options that
- affect the file mode, like fsGroup, and the result
- can be other mode bits set.'
- format: int32
- type: integer
- path:
- description: path is the relative path of the file
- to map the key to. May not be an absolute path.
- May not contain the path element '..'. May not start
- with the string '..'.
- type: string
- required:
- - key
- - path
- type: object
- type: array
- optional:
- description: optional field specify whether the Secret or
- its keys must be defined
- type: boolean
- secretName:
- description: 'secretName is the name of the secret in the
- pod''s namespace to use. More info: https://kubernetes.io/docs/concepts/storage/volumes#secret'
- type: string
- type: object
- storageos:
- description: storageOS represents a StorageOS volume attached
- and mounted on Kubernetes nodes.
- properties:
- fsType:
- description: fsType is the filesystem type to mount. Must
- be a filesystem type supported by the host operating system.
- Ex. "ext4", "xfs", "ntfs". Implicitly inferred to be "ext4"
- if unspecified.
- type: string
- readOnly:
- description: readOnly defaults to false (read/write). ReadOnly
- here will force the ReadOnly setting in VolumeMounts.
- type: boolean
- secretRef:
- description: secretRef specifies the secret to use for obtaining
- the StorageOS API credentials. If not specified, default
- values will be attempted.
- properties:
- name:
- description: 'Name of the referent. More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names
- TODO: Add other useful fields. apiVersion, kind, uid?'
- type: string
- type: object
- volumeName:
- description: volumeName is the human-readable name of the
- StorageOS volume. Volume names are only unique within
- a namespace.
- type: string
- volumeNamespace:
- description: volumeNamespace specifies the scope of the
- volume within StorageOS. If no namespace is specified
- then the Pod's namespace will be used. This allows the
- Kubernetes name scoping to be mirrored within StorageOS
- for tighter integration. Set VolumeName to any name to
- override the default behaviour. Set to "default" if you
- are not using namespaces within StorageOS. Namespaces
- that do not pre-exist within StorageOS will be created.
- type: string
- type: object
- vsphereVolume:
- description: vsphereVolume represents a vSphere volume attached
- and mounted on kubelets host machine
- properties:
- fsType:
- description: fsType is filesystem type to mount. Must be
- a filesystem type supported by the host operating system.
- Ex. "ext4", "xfs", "ntfs". Implicitly inferred to be "ext4"
- if unspecified.
- type: string
- storagePolicyID:
- description: storagePolicyID is the storage Policy Based
- Management (SPBM) profile ID associated with the StoragePolicyName.
- type: string
- storagePolicyName:
- description: storagePolicyName is the storage Policy Based
- Management (SPBM) profile name.
- type: string
- volumePath:
- description: volumePath is the path that identifies vSphere
- volume vmdk
- type: string
- required:
- - volumePath
- type: object
- required:
- - name
- type: object
- minItems: 1
- type: array
- x-kubernetes-preserve-unknown-fields: true
- required:
- - backupJobName
- - target
- - targetVolumeMounts
- - targetVolumes
- type: object
- status:
- description: RestoreJobStatus defines the observed state of RestoreJob
- properties:
- completionTimestamp:
- description: Date/time when the backup finished being processed.
- format: date-time
- type: string
- expiration:
- description: The date and time when the Backup is eligible for garbage
- collection. 'null' means the Backup is NOT be cleaned except delete
- manual.
- format: date-time
- type: string
- failureReason:
- description: Job failed reason.
- type: string
- phase:
- description: RestoreJobPhase The current phase. Valid values are New,
- InProgressPhy, InProgressLogic, Completed, Failed.
- enum:
- - New
- - InProgressPhy
- - InProgressLogic
- - Completed
- - Failed
- type: string
- startTimestamp:
- description: Date/time when the backup started being processed.
- format: date-time
- type: string
- type: object
- type: object
- served: true
- storage: true
- subresources:
- status: {}
diff --git a/config/crd/bases/dataprotection.kubeblocks.io_restores.yaml b/config/crd/bases/dataprotection.kubeblocks.io_restores.yaml
new file mode 100644
index 00000000000..a36a9f042d4
--- /dev/null
+++ b/config/crd/bases/dataprotection.kubeblocks.io_restores.yaml
@@ -0,0 +1,2522 @@
+apiVersion: apiextensions.k8s.io/v1
+kind: CustomResourceDefinition
+metadata:
+ annotations:
+ controller-gen.kubebuilder.io/version: v0.12.1
+ labels:
+ app.kubernetes.io/name: kubeblocks
+ name: restores.dataprotection.kubeblocks.io
+spec:
+ group: dataprotection.kubeblocks.io
+ names:
+ categories:
+ - kubeblocks
+ - all
+ kind: Restore
+ listKind: RestoreList
+ plural: restores
+ singular: restore
+ scope: Namespaced
+ versions:
+ - additionalPrinterColumns:
+ - jsonPath: .spec.backup.name
+ name: BACKUP
+ type: string
+ - description: Point in time for restoring
+ jsonPath: .spec.restoreTime
+ name: RESTORE-TIME
+ type: string
+ - description: Restore Status.
+ jsonPath: .status.phase
+ name: STATUS
+ type: string
+ - jsonPath: .status.duration
+ name: DURATION
+ type: string
+ - jsonPath: .metadata.creationTimestamp
+ name: CREATE-TIME
+ type: string
+ - jsonPath: .status.completionTimestamp
+ name: COMPLETION-TIME
+ type: string
+ name: v1alpha1
+ schema:
+ openAPIV3Schema:
+ description: Restore is the Schema for the restores API
+ properties:
+ apiVersion:
+ description: 'APIVersion defines the versioned schema of this representation
+ of an object. Servers should convert recognized schemas to the latest
+ internal value, and may reject unrecognized values. More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#resources'
+ type: string
+ kind:
+ description: 'Kind is a string value representing the REST resource this
+ object represents. Servers may infer this from the endpoint the client
+ submits requests to. Cannot be updated. In CamelCase. More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#types-kinds'
+ type: string
+ metadata:
+ type: object
+ spec:
+ description: RestoreSpec defines the desired state of Restore
+ properties:
+ backup:
+ description: 'backup name, the following behavior based on the backup
+ type: 1. Full: will be restored the full backup directly. 2. Incremental:
+ will be restored sequentially from the most recent full backup of
+ this incremental backup. 3. Differential: will be restored sequentially
+ from the parent backup of the differential backup. 4. Continuous:
+ will find the most recent full backup at this time point and the
+ input continuous backup to restore.'
+ properties:
+ name:
+ description: backup name
+ type: string
+ namespace:
+ description: backup namespace
+ type: string
+ required:
+ - name
+ - namespace
+ type: object
+ x-kubernetes-validations:
+ - message: forbidden to update spec.backupName
+ rule: self == oldSelf
+ containerResources:
+ description: specified the required resources of restore job's container.
+ properties:
+ claims:
+ description: "Claims lists the names of resources, defined in
+ spec.resourceClaims, that are used by this container. \n This
+ is an alpha field and requires enabling the DynamicResourceAllocation
+ feature gate. \n This field is immutable. It can only be set
+ for containers."
+ items:
+ description: ResourceClaim references one entry in PodSpec.ResourceClaims.
+ properties:
+ name:
+ description: Name must match the name of one entry in pod.spec.resourceClaims
+ of the Pod where this field is used. It makes that resource
+ available inside a container.
+ type: string
+ required:
+ - name
+ type: object
+ type: array
+ x-kubernetes-list-map-keys:
+ - name
+ x-kubernetes-list-type: map
+ limits:
+ additionalProperties:
+ anyOf:
+ - type: integer
+ - type: string
+ pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$
+ x-kubernetes-int-or-string: true
+ description: 'Limits describes the maximum amount of compute resources
+ allowed. More info: https://kubernetes.io/docs/concepts/configuration/manage-resources-containers/'
+ type: object
+ requests:
+ additionalProperties:
+ anyOf:
+ - type: integer
+ - type: string
+ pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$
+ x-kubernetes-int-or-string: true
+ description: 'Requests describes the minimum amount of compute
+ resources required. If Requests is omitted for a container,
+ it defaults to Limits if that is explicitly specified, otherwise
+ to an implementation-defined value. Requests cannot exceed Limits.
+ More info: https://kubernetes.io/docs/concepts/configuration/manage-resources-containers/'
+ type: object
+ type: object
+ env:
+ description: 'list of environment variables to set in the container
+ for restore and will be merged with the env of Backup and ActionSet.
+ the priority of merging is as follows: Restore env > Backup env
+ > ActionSet env.'
+ items:
+ description: EnvVar represents an environment variable present in
+ a Container.
+ properties:
+ name:
+ description: Name of the environment variable. Must be a C_IDENTIFIER.
+ type: string
+ value:
+ description: 'Variable references $(VAR_NAME) are expanded using
+ the previously defined environment variables in the container
+ and any service environment variables. If a variable cannot
+ be resolved, the reference in the input string will be unchanged.
+ Double $$ are reduced to a single $, which allows for escaping
+ the $(VAR_NAME) syntax: i.e. "$$(VAR_NAME)" will produce the
+ string literal "$(VAR_NAME)". Escaped references will never
+ be expanded, regardless of whether the variable exists or
+ not. Defaults to "".'
+ type: string
+ valueFrom:
+ description: Source for the environment variable's value. Cannot
+ be used if value is not empty.
+ properties:
+ configMapKeyRef:
+ description: Selects a key of a ConfigMap.
+ properties:
+ key:
+ description: The key to select.
+ type: string
+ name:
+ description: 'Name of the referent. More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names
+ TODO: Add other useful fields. apiVersion, kind, uid?'
+ type: string
+ optional:
+ description: Specify whether the ConfigMap or its key
+ must be defined
+ type: boolean
+ required:
+ - key
+ type: object
+ x-kubernetes-map-type: atomic
+ fieldRef:
+ description: 'Selects a field of the pod: supports metadata.name,
+ metadata.namespace, `metadata.labels['''']`, `metadata.annotations['''']`,
+ spec.nodeName, spec.serviceAccountName, status.hostIP,
+ status.podIP, status.podIPs.'
+ properties:
+ apiVersion:
+ description: Version of the schema the FieldPath is
+ written in terms of, defaults to "v1".
+ type: string
+ fieldPath:
+ description: Path of the field to select in the specified
+ API version.
+ type: string
+ required:
+ - fieldPath
+ type: object
+ x-kubernetes-map-type: atomic
+ resourceFieldRef:
+ description: 'Selects a resource of the container: only
+ resources limits and requests (limits.cpu, limits.memory,
+ limits.ephemeral-storage, requests.cpu, requests.memory
+ and requests.ephemeral-storage) are currently supported.'
+ properties:
+ containerName:
+ description: 'Container name: required for volumes,
+ optional for env vars'
+ type: string
+ divisor:
+ anyOf:
+ - type: integer
+ - type: string
+ description: Specifies the output format of the exposed
+ resources, defaults to "1"
+ pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$
+ x-kubernetes-int-or-string: true
+ resource:
+ description: 'Required: resource to select'
+ type: string
+ required:
+ - resource
+ type: object
+ x-kubernetes-map-type: atomic
+ secretKeyRef:
+ description: Selects a key of a secret in the pod's namespace
+ properties:
+ key:
+ description: The key of the secret to select from. Must
+ be a valid secret key.
+ type: string
+ name:
+ description: 'Name of the referent. More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names
+ TODO: Add other useful fields. apiVersion, kind, uid?'
+ type: string
+ optional:
+ description: Specify whether the Secret or its key must
+ be defined
+ type: boolean
+ required:
+ - key
+ type: object
+ x-kubernetes-map-type: atomic
+ type: object
+ required:
+ - name
+ type: object
+ type: array
+ x-kubernetes-preserve-unknown-fields: true
+ prepareDataConfig:
+ description: configuration for the action of "prepareData" phase,
+ including the persistent volume claims that need to be restored
+ and scheduling strategy of temporary recovery pod.
+ properties:
+ dataSourceRef:
+ description: dataSourceRef describes the configuration when using
+ `persistentVolumeClaim.spec.dataSourceRef` method for restoring.
+ it describes the source volume of the backup targetVolumes and
+ how to mount path in the restoring container.
+ properties:
+ mountPath:
+ description: mountPath path within the restoring container
+ at which the volume should be mounted.
+ type: string
+ volumeSource:
+ description: volumeSource describes the volume will be restored
+ from the specified volume of the backup targetVolumes. required
+ if the backup uses volume snapshot.
+ type: string
+ type: object
+ x-kubernetes-validations:
+ - message: at least one exists for volumeSource and mountPath.
+ rule: self.volumeSource != '' || self.mountPath !=''
+ - message: forbidden to update spec.prepareDataConfig.dataSourceRef
+ rule: self == oldSelf
+ schedulingSpec:
+ description: scheduling spec for restoring pod.
+ properties:
+ affinity:
+ description: affinity is a group of affinity scheduling rules.
+ refer to https://kubernetes.io/docs/concepts/configuration/assign-pod-node/
+ properties:
+ nodeAffinity:
+ description: Describes node affinity scheduling rules
+ for the pod.
+ properties:
+ preferredDuringSchedulingIgnoredDuringExecution:
+ description: The scheduler will prefer to schedule
+ pods to nodes that satisfy the affinity expressions
+ specified by this field, but it may choose a node
+ that violates one or more of the expressions. The
+ node that is most preferred is the one with the
+ greatest sum of weights, i.e. for each node that
+ meets all of the scheduling requirements (resource
+ request, requiredDuringScheduling affinity expressions,
+ etc.), compute a sum by iterating through the elements
+ of this field and adding "weight" to the sum if
+ the node matches the corresponding matchExpressions;
+ the node(s) with the highest sum are the most preferred.
+ items:
+ description: An empty preferred scheduling term
+ matches all objects with implicit weight 0 (i.e.
+ it's a no-op). A null preferred scheduling term
+ matches no objects (i.e. is also a no-op).
+ properties:
+ preference:
+ description: A node selector term, associated
+ with the corresponding weight.
+ properties:
+ matchExpressions:
+ description: A list of node selector requirements
+ by node's labels.
+ items:
+ description: A node selector requirement
+ is a selector that contains values,
+ a key, and an operator that relates
+ the key and values.
+ properties:
+ key:
+ description: The label key that the
+ selector applies to.
+ type: string
+ operator:
+ description: Represents a key's relationship
+ to a set of values. Valid operators
+ are In, NotIn, Exists, DoesNotExist.
+ Gt, and Lt.
+ type: string
+ values:
+ description: An array of string values.
+ If the operator is In or NotIn,
+ the values array must be non-empty.
+ If the operator is Exists or DoesNotExist,
+ the values array must be empty.
+ If the operator is Gt or Lt, the
+ values array must have a single
+ element, which will be interpreted
+ as an integer. This array is replaced
+ during a strategic merge patch.
+ items:
+ type: string
+ type: array
+ required:
+ - key
+ - operator
+ type: object
+ type: array
+ matchFields:
+ description: A list of node selector requirements
+ by node's fields.
+ items:
+ description: A node selector requirement
+ is a selector that contains values,
+ a key, and an operator that relates
+ the key and values.
+ properties:
+ key:
+ description: The label key that the
+ selector applies to.
+ type: string
+ operator:
+ description: Represents a key's relationship
+ to a set of values. Valid operators
+ are In, NotIn, Exists, DoesNotExist.
+ Gt, and Lt.
+ type: string
+ values:
+ description: An array of string values.
+ If the operator is In or NotIn,
+ the values array must be non-empty.
+ If the operator is Exists or DoesNotExist,
+ the values array must be empty.
+ If the operator is Gt or Lt, the
+ values array must have a single
+ element, which will be interpreted
+ as an integer. This array is replaced
+ during a strategic merge patch.
+ items:
+ type: string
+ type: array
+ required:
+ - key
+ - operator
+ type: object
+ type: array
+ type: object
+ x-kubernetes-map-type: atomic
+ weight:
+ description: Weight associated with matching
+ the corresponding nodeSelectorTerm, in the
+ range 1-100.
+ format: int32
+ type: integer
+ required:
+ - preference
+ - weight
+ type: object
+ type: array
+ requiredDuringSchedulingIgnoredDuringExecution:
+ description: If the affinity requirements specified
+ by this field are not met at scheduling time, the
+ pod will not be scheduled onto the node. If the
+ affinity requirements specified by this field cease
+ to be met at some point during pod execution (e.g.
+ due to an update), the system may or may not try
+ to eventually evict the pod from its node.
+ properties:
+ nodeSelectorTerms:
+ description: Required. A list of node selector
+ terms. The terms are ORed.
+ items:
+ description: A null or empty node selector term
+ matches no objects. The requirements of them
+ are ANDed. The TopologySelectorTerm type implements
+ a subset of the NodeSelectorTerm.
+ properties:
+ matchExpressions:
+ description: A list of node selector requirements
+ by node's labels.
+ items:
+ description: A node selector requirement
+ is a selector that contains values,
+ a key, and an operator that relates
+ the key and values.
+ properties:
+ key:
+ description: The label key that the
+ selector applies to.
+ type: string
+ operator:
+ description: Represents a key's relationship
+ to a set of values. Valid operators
+ are In, NotIn, Exists, DoesNotExist.
+ Gt, and Lt.
+ type: string
+ values:
+ description: An array of string values.
+ If the operator is In or NotIn,
+ the values array must be non-empty.
+ If the operator is Exists or DoesNotExist,
+ the values array must be empty.
+ If the operator is Gt or Lt, the
+ values array must have a single
+ element, which will be interpreted
+ as an integer. This array is replaced
+ during a strategic merge patch.
+ items:
+ type: string
+ type: array
+ required:
+ - key
+ - operator
+ type: object
+ type: array
+ matchFields:
+ description: A list of node selector requirements
+ by node's fields.
+ items:
+ description: A node selector requirement
+ is a selector that contains values,
+ a key, and an operator that relates
+ the key and values.
+ properties:
+ key:
+ description: The label key that the
+ selector applies to.
+ type: string
+ operator:
+ description: Represents a key's relationship
+ to a set of values. Valid operators
+ are In, NotIn, Exists, DoesNotExist.
+ Gt, and Lt.
+ type: string
+ values:
+ description: An array of string values.
+ If the operator is In or NotIn,
+ the values array must be non-empty.
+ If the operator is Exists or DoesNotExist,
+ the values array must be empty.
+ If the operator is Gt or Lt, the
+ values array must have a single
+ element, which will be interpreted
+ as an integer. This array is replaced
+ during a strategic merge patch.
+ items:
+ type: string
+ type: array
+ required:
+ - key
+ - operator
+ type: object
+ type: array
+ type: object
+ x-kubernetes-map-type: atomic
+ type: array
+ required:
+ - nodeSelectorTerms
+ type: object
+ x-kubernetes-map-type: atomic
+ type: object
+ podAffinity:
+ description: Describes pod affinity scheduling rules (e.g.
+ co-locate this pod in the same node, zone, etc. as some
+ other pod(s)).
+ properties:
+ preferredDuringSchedulingIgnoredDuringExecution:
+ description: The scheduler will prefer to schedule
+ pods to nodes that satisfy the affinity expressions
+ specified by this field, but it may choose a node
+ that violates one or more of the expressions. The
+ node that is most preferred is the one with the
+ greatest sum of weights, i.e. for each node that
+ meets all of the scheduling requirements (resource
+ request, requiredDuringScheduling affinity expressions,
+ etc.), compute a sum by iterating through the elements
+ of this field and adding "weight" to the sum if
+ the node has pods which matches the corresponding
+ podAffinityTerm; the node(s) with the highest sum
+ are the most preferred.
+ items:
+ description: The weights of all of the matched WeightedPodAffinityTerm
+ fields are added per-node to find the most preferred
+ node(s)
+ properties:
+ podAffinityTerm:
+ description: Required. A pod affinity term,
+ associated with the corresponding weight.
+ properties:
+ labelSelector:
+ description: A label query over a set of
+ resources, in this case pods.
+ properties:
+ matchExpressions:
+ description: matchExpressions is a list
+ of label selector requirements. The
+ requirements are ANDed.
+ items:
+ description: A label selector requirement
+ is a selector that contains values,
+ a key, and an operator that relates
+ the key and values.
+ properties:
+ key:
+ description: key is the label
+ key that the selector applies
+ to.
+ type: string
+ operator:
+ description: operator represents
+ a key's relationship to a set
+ of values. Valid operators are
+ In, NotIn, Exists and DoesNotExist.
+ type: string
+ values:
+ description: values is an array
+ of string values. If the operator
+ is In or NotIn, the values array
+ must be non-empty. If the operator
+ is Exists or DoesNotExist, the
+ values array must be empty.
+ This array is replaced during
+ a strategic merge patch.
+ items:
+ type: string
+ type: array
+ required:
+ - key
+ - operator
+ type: object
+ type: array
+ matchLabels:
+ additionalProperties:
+ type: string
+ description: matchLabels is a map of
+ {key,value} pairs. A single {key,value}
+ in the matchLabels map is equivalent
+ to an element of matchExpressions,
+ whose key field is "key", the operator
+ is "In", and the values array contains
+ only "value". The requirements are
+ ANDed.
+ type: object
+ type: object
+ x-kubernetes-map-type: atomic
+ namespaceSelector:
+ description: A label query over the set
+ of namespaces that the term applies to.
+ The term is applied to the union of the
+ namespaces selected by this field and
+ the ones listed in the namespaces field.
+ null selector and null or empty namespaces
+ list means "this pod's namespace". An
+ empty selector ({}) matches all namespaces.
+ properties:
+ matchExpressions:
+ description: matchExpressions is a list
+ of label selector requirements. The
+ requirements are ANDed.
+ items:
+ description: A label selector requirement
+ is a selector that contains values,
+ a key, and an operator that relates
+ the key and values.
+ properties:
+ key:
+ description: key is the label
+ key that the selector applies
+ to.
+ type: string
+ operator:
+ description: operator represents
+ a key's relationship to a set
+ of values. Valid operators are
+ In, NotIn, Exists and DoesNotExist.
+ type: string
+ values:
+ description: values is an array
+ of string values. If the operator
+ is In or NotIn, the values array
+ must be non-empty. If the operator
+ is Exists or DoesNotExist, the
+ values array must be empty.
+ This array is replaced during
+ a strategic merge patch.
+ items:
+ type: string
+ type: array
+ required:
+ - key
+ - operator
+ type: object
+ type: array
+ matchLabels:
+ additionalProperties:
+ type: string
+ description: matchLabels is a map of
+ {key,value} pairs. A single {key,value}
+ in the matchLabels map is equivalent
+ to an element of matchExpressions,
+ whose key field is "key", the operator
+ is "In", and the values array contains
+ only "value". The requirements are
+ ANDed.
+ type: object
+ type: object
+ x-kubernetes-map-type: atomic
+ namespaces:
+ description: namespaces specifies a static
+ list of namespace names that the term
+ applies to. The term is applied to the
+ union of the namespaces listed in this
+ field and the ones selected by namespaceSelector.
+ null or empty namespaces list and null
+ namespaceSelector means "this pod's namespace".
+ items:
+ type: string
+ type: array
+ topologyKey:
+ description: This pod should be co-located
+ (affinity) or not co-located (anti-affinity)
+ with the pods matching the labelSelector
+ in the specified namespaces, where co-located
+ is defined as running on a node whose
+ value of the label with key topologyKey
+ matches that of any node on which any
+ of the selected pods is running. Empty
+ topologyKey is not allowed.
+ type: string
+ required:
+ - topologyKey
+ type: object
+ weight:
+ description: weight associated with matching
+ the corresponding podAffinityTerm, in the
+ range 1-100.
+ format: int32
+ type: integer
+ required:
+ - podAffinityTerm
+ - weight
+ type: object
+ type: array
+ requiredDuringSchedulingIgnoredDuringExecution:
+ description: If the affinity requirements specified
+ by this field are not met at scheduling time, the
+ pod will not be scheduled onto the node. If the
+ affinity requirements specified by this field cease
+ to be met at some point during pod execution (e.g.
+ due to a pod label update), the system may or may
+ not try to eventually evict the pod from its node.
+ When there are multiple elements, the lists of nodes
+ corresponding to each podAffinityTerm are intersected,
+ i.e. all terms must be satisfied.
+ items:
+ description: Defines a set of pods (namely those
+ matching the labelSelector relative to the given
+ namespace(s)) that this pod should be co-located
+ (affinity) or not co-located (anti-affinity) with,
+ where co-located is defined as running on a node
+ whose value of the label with key
+ matches that of any node on which a pod of the
+ set of pods is running
+ properties:
+ labelSelector:
+ description: A label query over a set of resources,
+ in this case pods.
+ properties:
+ matchExpressions:
+ description: matchExpressions is a list
+ of label selector requirements. The requirements
+ are ANDed.
+ items:
+ description: A label selector requirement
+ is a selector that contains values,
+ a key, and an operator that relates
+ the key and values.
+ properties:
+ key:
+ description: key is the label key
+ that the selector applies to.
+ type: string
+ operator:
+ description: operator represents a
+ key's relationship to a set of values.
+ Valid operators are In, NotIn, Exists
+ and DoesNotExist.
+ type: string
+ values:
+ description: values is an array of
+ string values. If the operator is
+ In or NotIn, the values array must
+ be non-empty. If the operator is
+ Exists or DoesNotExist, the values
+ array must be empty. This array
+ is replaced during a strategic merge
+ patch.
+ items:
+ type: string
+ type: array
+ required:
+ - key
+ - operator
+ type: object
+ type: array
+ matchLabels:
+ additionalProperties:
+ type: string
+ description: matchLabels is a map of {key,value}
+ pairs. A single {key,value} in the matchLabels
+ map is equivalent to an element of matchExpressions,
+ whose key field is "key", the operator
+ is "In", and the values array contains
+ only "value". The requirements are ANDed.
+ type: object
+ type: object
+ x-kubernetes-map-type: atomic
+ namespaceSelector:
+ description: A label query over the set of namespaces
+ that the term applies to. The term is applied
+ to the union of the namespaces selected by
+ this field and the ones listed in the namespaces
+ field. null selector and null or empty namespaces
+ list means "this pod's namespace". An empty
+ selector ({}) matches all namespaces.
+ properties:
+ matchExpressions:
+ description: matchExpressions is a list
+ of label selector requirements. The requirements
+ are ANDed.
+ items:
+ description: A label selector requirement
+ is a selector that contains values,
+ a key, and an operator that relates
+ the key and values.
+ properties:
+ key:
+ description: key is the label key
+ that the selector applies to.
+ type: string
+ operator:
+ description: operator represents a
+ key's relationship to a set of values.
+ Valid operators are In, NotIn, Exists
+ and DoesNotExist.
+ type: string
+ values:
+ description: values is an array of
+ string values. If the operator is
+ In or NotIn, the values array must
+ be non-empty. If the operator is
+ Exists or DoesNotExist, the values
+ array must be empty. This array
+ is replaced during a strategic merge
+ patch.
+ items:
+ type: string
+ type: array
+ required:
+ - key
+ - operator
+ type: object
+ type: array
+ matchLabels:
+ additionalProperties:
+ type: string
+ description: matchLabels is a map of {key,value}
+ pairs. A single {key,value} in the matchLabels
+ map is equivalent to an element of matchExpressions,
+ whose key field is "key", the operator
+ is "In", and the values array contains
+ only "value". The requirements are ANDed.
+ type: object
+ type: object
+ x-kubernetes-map-type: atomic
+ namespaces:
+ description: namespaces specifies a static list
+ of namespace names that the term applies to.
+ The term is applied to the union of the namespaces
+ listed in this field and the ones selected
+ by namespaceSelector. null or empty namespaces
+ list and null namespaceSelector means "this
+ pod's namespace".
+ items:
+ type: string
+ type: array
+ topologyKey:
+ description: This pod should be co-located (affinity)
+ or not co-located (anti-affinity) with the
+ pods matching the labelSelector in the specified
+ namespaces, where co-located is defined as
+ running on a node whose value of the label
+ with key topologyKey matches that of any node
+ on which any of the selected pods is running.
+ Empty topologyKey is not allowed.
+ type: string
+ required:
+ - topologyKey
+ type: object
+ type: array
+ type: object
+ podAntiAffinity:
+ description: Describes pod anti-affinity scheduling rules
+ (e.g. avoid putting this pod in the same node, zone,
+ etc. as some other pod(s)).
+ properties:
+ preferredDuringSchedulingIgnoredDuringExecution:
+ description: The scheduler will prefer to schedule
+ pods to nodes that satisfy the anti-affinity expressions
+ specified by this field, but it may choose a node
+ that violates one or more of the expressions. The
+ node that is most preferred is the one with the
+ greatest sum of weights, i.e. for each node that
+ meets all of the scheduling requirements (resource
+ request, requiredDuringScheduling anti-affinity
+ expressions, etc.), compute a sum by iterating through
+ the elements of this field and adding "weight" to
+ the sum if the node has pods which matches the corresponding
+ podAffinityTerm; the node(s) with the highest sum
+ are the most preferred.
+ items:
+ description: The weights of all of the matched WeightedPodAffinityTerm
+ fields are added per-node to find the most preferred
+ node(s)
+ properties:
+ podAffinityTerm:
+ description: Required. A pod affinity term,
+ associated with the corresponding weight.
+ properties:
+ labelSelector:
+ description: A label query over a set of
+ resources, in this case pods.
+ properties:
+ matchExpressions:
+ description: matchExpressions is a list
+ of label selector requirements. The
+ requirements are ANDed.
+ items:
+ description: A label selector requirement
+ is a selector that contains values,
+ a key, and an operator that relates
+ the key and values.
+ properties:
+ key:
+ description: key is the label
+ key that the selector applies
+ to.
+ type: string
+ operator:
+ description: operator represents
+ a key's relationship to a set
+ of values. Valid operators are
+ In, NotIn, Exists and DoesNotExist.
+ type: string
+ values:
+ description: values is an array
+ of string values. If the operator
+ is In or NotIn, the values array
+ must be non-empty. If the operator
+ is Exists or DoesNotExist, the
+ values array must be empty.
+ This array is replaced during
+ a strategic merge patch.
+ items:
+ type: string
+ type: array
+ required:
+ - key
+ - operator
+ type: object
+ type: array
+ matchLabels:
+ additionalProperties:
+ type: string
+ description: matchLabels is a map of
+ {key,value} pairs. A single {key,value}
+ in the matchLabels map is equivalent
+ to an element of matchExpressions,
+ whose key field is "key", the operator
+ is "In", and the values array contains
+ only "value". The requirements are
+ ANDed.
+ type: object
+ type: object
+ x-kubernetes-map-type: atomic
+ namespaceSelector:
+ description: A label query over the set
+ of namespaces that the term applies to.
+ The term is applied to the union of the
+ namespaces selected by this field and
+ the ones listed in the namespaces field.
+ null selector and null or empty namespaces
+ list means "this pod's namespace". An
+ empty selector ({}) matches all namespaces.
+ properties:
+ matchExpressions:
+ description: matchExpressions is a list
+ of label selector requirements. The
+ requirements are ANDed.
+ items:
+ description: A label selector requirement
+ is a selector that contains values,
+ a key, and an operator that relates
+ the key and values.
+ properties:
+ key:
+ description: key is the label
+ key that the selector applies
+ to.
+ type: string
+ operator:
+ description: operator represents
+ a key's relationship to a set
+ of values. Valid operators are
+ In, NotIn, Exists and DoesNotExist.
+ type: string
+ values:
+ description: values is an array
+ of string values. If the operator
+ is In or NotIn, the values array
+ must be non-empty. If the operator
+ is Exists or DoesNotExist, the
+ values array must be empty.
+ This array is replaced during
+ a strategic merge patch.
+ items:
+ type: string
+ type: array
+ required:
+ - key
+ - operator
+ type: object
+ type: array
+ matchLabels:
+ additionalProperties:
+ type: string
+ description: matchLabels is a map of
+ {key,value} pairs. A single {key,value}
+ in the matchLabels map is equivalent
+ to an element of matchExpressions,
+ whose key field is "key", the operator
+ is "In", and the values array contains
+ only "value". The requirements are
+ ANDed.
+ type: object
+ type: object
+ x-kubernetes-map-type: atomic
+ namespaces:
+ description: namespaces specifies a static
+ list of namespace names that the term
+ applies to. The term is applied to the
+ union of the namespaces listed in this
+ field and the ones selected by namespaceSelector.
+ null or empty namespaces list and null
+ namespaceSelector means "this pod's namespace".
+ items:
+ type: string
+ type: array
+ topologyKey:
+ description: This pod should be co-located
+ (affinity) or not co-located (anti-affinity)
+ with the pods matching the labelSelector
+ in the specified namespaces, where co-located
+ is defined as running on a node whose
+ value of the label with key topologyKey
+ matches that of any node on which any
+ of the selected pods is running. Empty
+ topologyKey is not allowed.
+ type: string
+ required:
+ - topologyKey
+ type: object
+ weight:
+ description: weight associated with matching
+ the corresponding podAffinityTerm, in the
+ range 1-100.
+ format: int32
+ type: integer
+ required:
+ - podAffinityTerm
+ - weight
+ type: object
+ type: array
+ requiredDuringSchedulingIgnoredDuringExecution:
+ description: If the anti-affinity requirements specified
+ by this field are not met at scheduling time, the
+ pod will not be scheduled onto the node. If the
+ anti-affinity requirements specified by this field
+ cease to be met at some point during pod execution
+ (e.g. due to a pod label update), the system may
+ or may not try to eventually evict the pod from
+ its node. When there are multiple elements, the
+ lists of nodes corresponding to each podAffinityTerm
+ are intersected, i.e. all terms must be satisfied.
+ items:
+ description: Defines a set of pods (namely those
+ matching the labelSelector relative to the given
+ namespace(s)) that this pod should be co-located
+ (affinity) or not co-located (anti-affinity) with,
+ where co-located is defined as running on a node
+ whose value of the label with key
+ matches that of any node on which a pod of the
+ set of pods is running
+ properties:
+ labelSelector:
+ description: A label query over a set of resources,
+ in this case pods.
+ properties:
+ matchExpressions:
+ description: matchExpressions is a list
+ of label selector requirements. The requirements
+ are ANDed.
+ items:
+ description: A label selector requirement
+ is a selector that contains values,
+ a key, and an operator that relates
+ the key and values.
+ properties:
+ key:
+ description: key is the label key
+ that the selector applies to.
+ type: string
+ operator:
+ description: operator represents a
+ key's relationship to a set of values.
+ Valid operators are In, NotIn, Exists
+ and DoesNotExist.
+ type: string
+ values:
+ description: values is an array of
+ string values. If the operator is
+ In or NotIn, the values array must
+ be non-empty. If the operator is
+ Exists or DoesNotExist, the values
+ array must be empty. This array
+ is replaced during a strategic merge
+ patch.
+ items:
+ type: string
+ type: array
+ required:
+ - key
+ - operator
+ type: object
+ type: array
+ matchLabels:
+ additionalProperties:
+ type: string
+ description: matchLabels is a map of {key,value}
+ pairs. A single {key,value} in the matchLabels
+ map is equivalent to an element of matchExpressions,
+ whose key field is "key", the operator
+ is "In", and the values array contains
+ only "value". The requirements are ANDed.
+ type: object
+ type: object
+ x-kubernetes-map-type: atomic
+ namespaceSelector:
+ description: A label query over the set of namespaces
+ that the term applies to. The term is applied
+ to the union of the namespaces selected by
+ this field and the ones listed in the namespaces
+ field. null selector and null or empty namespaces
+ list means "this pod's namespace". An empty
+ selector ({}) matches all namespaces.
+ properties:
+ matchExpressions:
+ description: matchExpressions is a list
+ of label selector requirements. The requirements
+ are ANDed.
+ items:
+ description: A label selector requirement
+ is a selector that contains values,
+ a key, and an operator that relates
+ the key and values.
+ properties:
+ key:
+ description: key is the label key
+ that the selector applies to.
+ type: string
+ operator:
+ description: operator represents a
+ key's relationship to a set of values.
+ Valid operators are In, NotIn, Exists
+ and DoesNotExist.
+ type: string
+ values:
+ description: values is an array of
+ string values. If the operator is
+ In or NotIn, the values array must
+ be non-empty. If the operator is
+ Exists or DoesNotExist, the values
+ array must be empty. This array
+ is replaced during a strategic merge
+ patch.
+ items:
+ type: string
+ type: array
+ required:
+ - key
+ - operator
+ type: object
+ type: array
+ matchLabels:
+ additionalProperties:
+ type: string
+ description: matchLabels is a map of {key,value}
+ pairs. A single {key,value} in the matchLabels
+ map is equivalent to an element of matchExpressions,
+ whose key field is "key", the operator
+ is "In", and the values array contains
+ only "value". The requirements are ANDed.
+ type: object
+ type: object
+ x-kubernetes-map-type: atomic
+ namespaces:
+ description: namespaces specifies a static list
+ of namespace names that the term applies to.
+ The term is applied to the union of the namespaces
+ listed in this field and the ones selected
+ by namespaceSelector. null or empty namespaces
+ list and null namespaceSelector means "this
+ pod's namespace".
+ items:
+ type: string
+ type: array
+ topologyKey:
+ description: This pod should be co-located (affinity)
+ or not co-located (anti-affinity) with the
+ pods matching the labelSelector in the specified
+ namespaces, where co-located is defined as
+ running on a node whose value of the label
+ with key topologyKey matches that of any node
+ on which any of the selected pods is running.
+ Empty topologyKey is not allowed.
+ type: string
+ required:
+ - topologyKey
+ type: object
+ type: array
+ type: object
+ type: object
+ nodeName:
+ description: nodeName is a request to schedule this pod onto
+ a specific node. If it is non-empty, the scheduler simply
+ schedules this pod onto that node, assuming that it fits
+ resource requirements.
+ type: string
+ nodeSelector:
+ additionalProperties:
+ type: string
+ description: 'nodeSelector is a selector which must be true
+ for the pod to fit on a node. Selector which must match
+ a node''s labels for the pod to be scheduled on that node.
+ More info: https://kubernetes.io/docs/concepts/configuration/assign-pod-node/'
+ type: object
+ x-kubernetes-map-type: atomic
+ schedulerName:
+ description: If specified, the pod will be dispatched by specified
+ scheduler. If not specified, the pod will be dispatched
+ by default scheduler.
+ type: string
+ tolerations:
+ description: the restoring pod's tolerations.
+ items:
+ description: The pod this Toleration is attached to tolerates
+ any taint that matches the triple using
+ the matching operator .
+ properties:
+ effect:
+ description: Effect indicates the taint effect to match.
+ Empty means match all taint effects. When specified,
+ allowed values are NoSchedule, PreferNoSchedule and
+ NoExecute.
+ type: string
+ key:
+ description: Key is the taint key that the toleration
+ applies to. Empty means match all taint keys. If the
+ key is empty, operator must be Exists; this combination
+ means to match all values and all keys.
+ type: string
+ operator:
+ description: Operator represents a key's relationship
+ to the value. Valid operators are Exists and Equal.
+ Defaults to Equal. Exists is equivalent to wildcard
+ for value, so that a pod can tolerate all taints of
+ a particular category.
+ type: string
+ tolerationSeconds:
+ description: TolerationSeconds represents the period
+ of time the toleration (which must be of effect NoExecute,
+ otherwise this field is ignored) tolerates the taint.
+ By default, it is not set, which means tolerate the
+ taint forever (do not evict). Zero and negative values
+ will be treated as 0 (evict immediately) by the system.
+ format: int64
+ type: integer
+ value:
+ description: Value is the taint value the toleration
+ matches to. If the operator is Exists, the value should
+ be empty, otherwise just a regular string.
+ type: string
+ type: object
+ type: array
+ topologySpreadConstraints:
+ description: topologySpreadConstraints describes how a group
+ of pods ought to spread across topology domains. Scheduler
+ will schedule pods in a way which abides by the constraints.
+ refer to https://kubernetes.io/docs/concepts/scheduling-eviction/topology-spread-constraints/
+ items:
+ description: TopologySpreadConstraint specifies how to spread
+ matching pods among the given topology.
+ properties:
+ labelSelector:
+ description: LabelSelector is used to find matching
+ pods. Pods that match this label selector are counted
+ to determine the number of pods in their corresponding
+ topology domain.
+ properties:
+ matchExpressions:
+ description: matchExpressions is a list of label
+ selector requirements. The requirements are ANDed.
+ items:
+ description: A label selector requirement is a
+ selector that contains values, a key, and an
+ operator that relates the key and values.
+ properties:
+ key:
+ description: key is the label key that the
+ selector applies to.
+ type: string
+ operator:
+ description: operator represents a key's relationship
+ to a set of values. Valid operators are
+ In, NotIn, Exists and DoesNotExist.
+ type: string
+ values:
+ description: values is an array of string
+ values. If the operator is In or NotIn,
+ the values array must be non-empty. If the
+ operator is Exists or DoesNotExist, the
+ values array must be empty. This array is
+ replaced during a strategic merge patch.
+ items:
+ type: string
+ type: array
+ required:
+ - key
+ - operator
+ type: object
+ type: array
+ matchLabels:
+ additionalProperties:
+ type: string
+ description: matchLabels is a map of {key,value}
+ pairs. A single {key,value} in the matchLabels
+ map is equivalent to an element of matchExpressions,
+ whose key field is "key", the operator is "In",
+ and the values array contains only "value". The
+ requirements are ANDed.
+ type: object
+ type: object
+ x-kubernetes-map-type: atomic
+ matchLabelKeys:
+ description: "MatchLabelKeys is a set of pod label keys
+ to select the pods over which spreading will be calculated.
+ The keys are used to lookup values from the incoming
+ pod labels, those key-value labels are ANDed with
+ labelSelector to select the group of existing pods
+ over which spreading will be calculated for the incoming
+ pod. The same key is forbidden to exist in both MatchLabelKeys
+ and LabelSelector. MatchLabelKeys cannot be set when
+ LabelSelector isn't set. Keys that don't exist in
+ the incoming pod labels will be ignored. A null or
+ empty list means only match against labelSelector.
+ \n This is a beta field and requires the MatchLabelKeysInPodTopologySpread
+ feature gate to be enabled (enabled by default)."
+ items:
+ type: string
+ type: array
+ x-kubernetes-list-type: atomic
+ maxSkew:
+ description: 'MaxSkew describes the degree to which
+ pods may be unevenly distributed. When `whenUnsatisfiable=DoNotSchedule`,
+ it is the maximum permitted difference between the
+ number of matching pods in the target topology and
+ the global minimum. The global minimum is the minimum
+ number of matching pods in an eligible domain or zero
+ if the number of eligible domains is less than MinDomains.
+ For example, in a 3-zone cluster, MaxSkew is set to
+ 1, and pods with the same labelSelector spread as
+ 2/2/1: In this case, the global minimum is 1. | zone1
+ | zone2 | zone3 | | P P | P P | P | - if MaxSkew
+ is 1, incoming pod can only be scheduled to zone3
+ to become 2/2/2; scheduling it onto zone1(zone2) would
+ make the ActualSkew(3-1) on zone1(zone2) violate MaxSkew(1).
+ - if MaxSkew is 2, incoming pod can be scheduled onto
+ any zone. When `whenUnsatisfiable=ScheduleAnyway`,
+ it is used to give higher precedence to topologies
+ that satisfy it. It''s a required field. Default value
+ is 1 and 0 is not allowed.'
+ format: int32
+ type: integer
+ minDomains:
+ description: "MinDomains indicates a minimum number
+ of eligible domains. When the number of eligible domains
+ with matching topology keys is less than minDomains,
+ Pod Topology Spread treats \"global minimum\" as 0,
+ and then the calculation of Skew is performed. And
+ when the number of eligible domains with matching
+ topology keys equals or greater than minDomains, this
+ value has no effect on scheduling. As a result, when
+ the number of eligible domains is less than minDomains,
+ scheduler won't schedule more than maxSkew Pods to
+ those domains. If value is nil, the constraint behaves
+ as if MinDomains is equal to 1. Valid values are integers
+ greater than 0. When value is not nil, WhenUnsatisfiable
+ must be DoNotSchedule. \n For example, in a 3-zone
+ cluster, MaxSkew is set to 2, MinDomains is set to
+ 5 and pods with the same labelSelector spread as 2/2/2:
+ | zone1 | zone2 | zone3 | | P P | P P | P P |
+ The number of domains is less than 5(MinDomains),
+ so \"global minimum\" is treated as 0. In this situation,
+ new pod with the same labelSelector cannot be scheduled,
+ because computed skew will be 3(3 - 0) if new Pod
+ is scheduled to any of the three zones, it will violate
+ MaxSkew. \n This is a beta field and requires the
+ MinDomainsInPodTopologySpread feature gate to be enabled
+ (enabled by default)."
+ format: int32
+ type: integer
+ nodeAffinityPolicy:
+ description: "NodeAffinityPolicy indicates how we will
+ treat Pod's nodeAffinity/nodeSelector when calculating
+ pod topology spread skew. Options are: - Honor: only
+ nodes matching nodeAffinity/nodeSelector are included
+ in the calculations. - Ignore: nodeAffinity/nodeSelector
+ are ignored. All nodes are included in the calculations.
+ \n If this value is nil, the behavior is equivalent
+ to the Honor policy. This is a beta-level feature
+ default enabled by the NodeInclusionPolicyInPodTopologySpread
+ feature flag."
+ type: string
+ nodeTaintsPolicy:
+ description: "NodeTaintsPolicy indicates how we will
+ treat node taints when calculating pod topology spread
+ skew. Options are: - Honor: nodes without taints,
+ along with tainted nodes for which the incoming pod
+ has a toleration, are included. - Ignore: node taints
+ are ignored. All nodes are included. \n If this value
+ is nil, the behavior is equivalent to the Ignore policy.
+ This is a beta-level feature default enabled by the
+ NodeInclusionPolicyInPodTopologySpread feature flag."
+ type: string
+ topologyKey:
+ description: TopologyKey is the key of node labels.
+ Nodes that have a label with this key and identical
+ values are considered to be in the same topology.
+ We consider each as a "bucket", and try
+ to put balanced number of pods into each bucket. We
+ define a domain as a particular instance of a topology.
+ Also, we define an eligible domain as a domain whose
+ nodes meet the requirements of nodeAffinityPolicy
+ and nodeTaintsPolicy. e.g. If TopologyKey is "kubernetes.io/hostname",
+ each Node is a domain of that topology. And, if TopologyKey
+ is "topology.kubernetes.io/zone", each zone is a domain
+ of that topology. It's a required field.
+ type: string
+ whenUnsatisfiable:
+ description: 'WhenUnsatisfiable indicates how to deal
+ with a pod if it doesn''t satisfy the spread constraint.
+ - DoNotSchedule (default) tells the scheduler not
+ to schedule it. - ScheduleAnyway tells the scheduler
+ to schedule the pod in any location, but giving higher
+ precedence to topologies that would help reduce the
+ skew. A constraint is considered "Unsatisfiable" for
+ an incoming pod if and only if every possible node
+ assignment for that pod would violate "MaxSkew" on
+ some topology. For example, in a 3-zone cluster, MaxSkew
+ is set to 1, and pods with the same labelSelector
+ spread as 3/1/1: | zone1 | zone2 | zone3 | | P P P
+ | P | P | If WhenUnsatisfiable is set to DoNotSchedule,
+ incoming pod can only be scheduled to zone2(zone3)
+ to become 3/2/1(3/1/2) as ActualSkew(2-1) on zone2(zone3)
+ satisfies MaxSkew(1). In other words, the cluster
+ can still be imbalanced, but scheduler won''t make
+ it *more* imbalanced. It''s a required field.'
+ type: string
+ required:
+ - maxSkew
+ - topologyKey
+ - whenUnsatisfiable
+ type: object
+ type: array
+ type: object
+ x-kubernetes-validations:
+ - message: forbidden to update spec.prepareDataConfig.schedulingSpec
+ rule: self == oldSelf
+ volumeClaimManagementPolicy:
+ default: Parallel
+ description: 'VolumeClaimManagementPolicy defines recovery strategy
+ for persistent volume claim. supported policies are as follows:
+ 1. Parallel: parallel recovery of persistent volume claim. 2.
+ Serial: restore the persistent volume claim in sequence, and
+ wait until the previous persistent volume claim is restored
+ before restoring a new one.'
+ enum:
+ - Parallel
+ - Serial
+ type: string
+ volumeClaims:
+ description: volumeClaims defines the persistent Volume claims
+ that need to be restored and mount them together into the restore
+ job. these persistent Volume claims will be created if not exist.
+ items:
+ properties:
+ metadata:
+ description: 'Standard object''s metadata. More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#metadata'
+ properties:
+ annotations:
+ additionalProperties:
+ type: string
+ type: object
+ finalizers:
+ items:
+ type: string
+ type: array
+ labels:
+ additionalProperties:
+ type: string
+ type: object
+ name:
+ type: string
+ namespace:
+ type: string
+ type: object
+ mountPath:
+ description: mountPath path within the restoring container
+ at which the volume should be mounted.
+ type: string
+ volumeClaimSpec:
+ description: volumeClaimSpec defines the desired characteristics
+ of a persistent volume claim.
+ properties:
+ accessModes:
+ description: 'accessModes contains the desired access
+ modes the volume should have. More info: https://kubernetes.io/docs/concepts/storage/persistent-volumes#access-modes-1'
+ items:
+ type: string
+ type: array
+ dataSource:
+ description: 'dataSource field can be used to specify
+ either: * An existing VolumeSnapshot object (snapshot.storage.k8s.io/VolumeSnapshot)
+ * An existing PVC (PersistentVolumeClaim) If the provisioner
+ or an external controller can support the specified
+ data source, it will create a new volume based on
+ the contents of the specified data source. When the
+ AnyVolumeDataSource feature gate is enabled, dataSource
+ contents will be copied to dataSourceRef, and dataSourceRef
+ contents will be copied to dataSource when dataSourceRef.namespace
+ is not specified. If the namespace is specified, then
+ dataSourceRef will not be copied to dataSource.'
+ properties:
+ apiGroup:
+ description: APIGroup is the group for the resource
+ being referenced. If APIGroup is not specified,
+ the specified Kind must be in the core API group.
+ For any other third-party types, APIGroup is required.
+ type: string
+ kind:
+ description: Kind is the type of resource being
+ referenced
+ type: string
+ name:
+ description: Name is the name of resource being
+ referenced
+ type: string
+ required:
+ - kind
+ - name
+ type: object
+ x-kubernetes-map-type: atomic
+ dataSourceRef:
+ description: 'dataSourceRef specifies the object from
+ which to populate the volume with data, if a non-empty
+ volume is desired. This may be any object from a non-empty
+ API group (non core object) or a PersistentVolumeClaim
+ object. When this field is specified, volume binding
+ will only succeed if the type of the specified object
+ matches some installed volume populator or dynamic
+ provisioner. This field will replace the functionality
+ of the dataSource field and as such if both fields
+ are non-empty, they must have the same value. For
+ backwards compatibility, when namespace isn''t specified
+ in dataSourceRef, both fields (dataSource and dataSourceRef)
+ will be set to the same value automatically if one
+ of them is empty and the other is non-empty. When
+ namespace is specified in dataSourceRef, dataSource
+ isn''t set to the same value and must be empty. There
+ are three important differences between dataSource
+ and dataSourceRef: * While dataSource only allows
+ two specific types of objects, dataSourceRef allows
+ any non-core object, as well as PersistentVolumeClaim
+ objects. * While dataSource ignores disallowed values
+ (dropping them), dataSourceRef preserves all values,
+ and generates an error if a disallowed value is specified.
+ * While dataSource only allows local objects, dataSourceRef
+ allows objects in any namespaces. (Beta) Using this
+ field requires the AnyVolumeDataSource feature gate
+ to be enabled. (Alpha) Using the namespace field of
+ dataSourceRef requires the CrossNamespaceVolumeDataSource
+ feature gate to be enabled.'
+ properties:
+ apiGroup:
+ description: APIGroup is the group for the resource
+ being referenced. If APIGroup is not specified,
+ the specified Kind must be in the core API group.
+ For any other third-party types, APIGroup is required.
+ type: string
+ kind:
+ description: Kind is the type of resource being
+ referenced
+ type: string
+ name:
+ description: Name is the name of resource being
+ referenced
+ type: string
+ namespace:
+ description: Namespace is the namespace of resource
+ being referenced Note that when a namespace is
+ specified, a gateway.networking.k8s.io/ReferenceGrant
+ object is required in the referent namespace to
+ allow that namespace's owner to accept the reference.
+ See the ReferenceGrant documentation for details.
+ (Alpha) This field requires the CrossNamespaceVolumeDataSource
+ feature gate to be enabled.
+ type: string
+ required:
+ - kind
+ - name
+ type: object
+ resources:
+ description: 'resources represents the minimum resources
+ the volume should have. If RecoverVolumeExpansionFailure
+ feature is enabled users are allowed to specify resource
+ requirements that are lower than previous value but
+ must still be higher than capacity recorded in the
+ status field of the claim. More info: https://kubernetes.io/docs/concepts/storage/persistent-volumes#resources'
+ properties:
+ claims:
+ description: "Claims lists the names of resources,
+ defined in spec.resourceClaims, that are used
+ by this container. \n This is an alpha field and
+ requires enabling the DynamicResourceAllocation
+ feature gate. \n This field is immutable. It can
+ only be set for containers."
+ items:
+ description: ResourceClaim references one entry
+ in PodSpec.ResourceClaims.
+ properties:
+ name:
+ description: Name must match the name of one
+ entry in pod.spec.resourceClaims of the
+ Pod where this field is used. It makes that
+ resource available inside a container.
+ type: string
+ required:
+ - name
+ type: object
+ type: array
+ x-kubernetes-list-map-keys:
+ - name
+ x-kubernetes-list-type: map
+ limits:
+ additionalProperties:
+ anyOf:
+ - type: integer
+ - type: string
+ pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$
+ x-kubernetes-int-or-string: true
+ description: 'Limits describes the maximum amount
+ of compute resources allowed. More info: https://kubernetes.io/docs/concepts/configuration/manage-resources-containers/'
+ type: object
+ requests:
+ additionalProperties:
+ anyOf:
+ - type: integer
+ - type: string
+ pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$
+ x-kubernetes-int-or-string: true
+ description: 'Requests describes the minimum amount
+ of compute resources required. If Requests is
+ omitted for a container, it defaults to Limits
+ if that is explicitly specified, otherwise to
+ an implementation-defined value. Requests cannot
+ exceed Limits. More info: https://kubernetes.io/docs/concepts/configuration/manage-resources-containers/'
+ type: object
+ type: object
+ selector:
+ description: selector is a label query over volumes
+ to consider for binding.
+ properties:
+ matchExpressions:
+ description: matchExpressions is a list of label
+ selector requirements. The requirements are ANDed.
+ items:
+ description: A label selector requirement is a
+ selector that contains values, a key, and an
+ operator that relates the key and values.
+ properties:
+ key:
+ description: key is the label key that the
+ selector applies to.
+ type: string
+ operator:
+ description: operator represents a key's relationship
+ to a set of values. Valid operators are
+ In, NotIn, Exists and DoesNotExist.
+ type: string
+ values:
+ description: values is an array of string
+ values. If the operator is In or NotIn,
+ the values array must be non-empty. If the
+ operator is Exists or DoesNotExist, the
+ values array must be empty. This array is
+ replaced during a strategic merge patch.
+ items:
+ type: string
+ type: array
+ required:
+ - key
+ - operator
+ type: object
+ type: array
+ matchLabels:
+ additionalProperties:
+ type: string
+ description: matchLabels is a map of {key,value}
+ pairs. A single {key,value} in the matchLabels
+ map is equivalent to an element of matchExpressions,
+ whose key field is "key", the operator is "In",
+ and the values array contains only "value". The
+ requirements are ANDed.
+ type: object
+ type: object
+ x-kubernetes-map-type: atomic
+ storageClassName:
+ description: 'storageClassName is the name of the StorageClass
+ required by the claim. More info: https://kubernetes.io/docs/concepts/storage/persistent-volumes#class-1'
+ type: string
+ volumeMode:
+ description: volumeMode defines what type of volume
+ is required by the claim. Value of Filesystem is implied
+ when not included in claim spec.
+ type: string
+ volumeName:
+ description: volumeName is the binding reference to
+ the PersistentVolume backing this claim.
+ type: string
+ type: object
+ volumeSource:
+ description: volumeSource describes the volume will be restored
+ from the specified volume of the backup targetVolumes.
+ required if the backup uses volume snapshot.
+ type: string
+ required:
+ - metadata
+ - volumeClaimSpec
+ type: object
+ x-kubernetes-validations:
+ - message: at least one exists for volumeSource and mountPath.
+ rule: self.volumeSource != '' || self.mountPath !=''
+ type: array
+ x-kubernetes-validations:
+ - message: forbidden to update spec.prepareDataConfig.volumeClaims
+ rule: self == oldSelf
+ volumeClaimsTemplate:
+ description: volumeClaimsTemplate defines a template to build
+ persistent Volume claims that need to be restored. these claims
+ will be created in an orderly manner based on the number of
+ replicas or reused if already exist.
+ properties:
+ replicas:
+ description: the replicas of persistent volume claim which
+ need to be created and restored. the format of created claim
+ name is "-".
+ format: int32
+ minimum: 1
+ type: integer
+ startingIndex:
+ description: the starting index for the created persistent
+ volume claim by according to template. minimum is 0.
+ format: int32
+ minimum: 0
+ type: integer
+ templates:
+ description: templates is a list of volume claims.
+ items:
+ properties:
+ metadata:
+ description: 'Standard object''s metadata. More info:
+ https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#metadata'
+ properties:
+ annotations:
+ additionalProperties:
+ type: string
+ type: object
+ finalizers:
+ items:
+ type: string
+ type: array
+ labels:
+ additionalProperties:
+ type: string
+ type: object
+ name:
+ type: string
+ namespace:
+ type: string
+ type: object
+ mountPath:
+ description: mountPath path within the restoring container
+ at which the volume should be mounted.
+ type: string
+ volumeClaimSpec:
+ description: volumeClaimSpec defines the desired characteristics
+ of a persistent volume claim.
+ properties:
+ accessModes:
+ description: 'accessModes contains the desired access
+ modes the volume should have. More info: https://kubernetes.io/docs/concepts/storage/persistent-volumes#access-modes-1'
+ items:
+ type: string
+ type: array
+ dataSource:
+ description: 'dataSource field can be used to specify
+ either: * An existing VolumeSnapshot object (snapshot.storage.k8s.io/VolumeSnapshot)
+ * An existing PVC (PersistentVolumeClaim) If the
+ provisioner or an external controller can support
+ the specified data source, it will create a new
+ volume based on the contents of the specified
+ data source. When the AnyVolumeDataSource feature
+ gate is enabled, dataSource contents will be copied
+ to dataSourceRef, and dataSourceRef contents will
+ be copied to dataSource when dataSourceRef.namespace
+ is not specified. If the namespace is specified,
+ then dataSourceRef will not be copied to dataSource.'
+ properties:
+ apiGroup:
+ description: APIGroup is the group for the resource
+ being referenced. If APIGroup is not specified,
+ the specified Kind must be in the core API
+ group. For any other third-party types, APIGroup
+ is required.
+ type: string
+ kind:
+ description: Kind is the type of resource being
+ referenced
+ type: string
+ name:
+ description: Name is the name of resource being
+ referenced
+ type: string
+ required:
+ - kind
+ - name
+ type: object
+ x-kubernetes-map-type: atomic
+ dataSourceRef:
+ description: 'dataSourceRef specifies the object
+ from which to populate the volume with data, if
+ a non-empty volume is desired. This may be any
+ object from a non-empty API group (non core object)
+ or a PersistentVolumeClaim object. When this field
+ is specified, volume binding will only succeed
+ if the type of the specified object matches some
+ installed volume populator or dynamic provisioner.
+ This field will replace the functionality of the
+ dataSource field and as such if both fields are
+ non-empty, they must have the same value. For
+ backwards compatibility, when namespace isn''t
+ specified in dataSourceRef, both fields (dataSource
+ and dataSourceRef) will be set to the same value
+ automatically if one of them is empty and the
+ other is non-empty. When namespace is specified
+ in dataSourceRef, dataSource isn''t set to the
+ same value and must be empty. There are three
+ important differences between dataSource and dataSourceRef:
+ * While dataSource only allows two specific types
+ of objects, dataSourceRef allows any non-core
+ object, as well as PersistentVolumeClaim objects.
+ * While dataSource ignores disallowed values (dropping
+ them), dataSourceRef preserves all values, and
+ generates an error if a disallowed value is specified.
+ * While dataSource only allows local objects,
+ dataSourceRef allows objects in any namespaces.
+ (Beta) Using this field requires the AnyVolumeDataSource
+ feature gate to be enabled. (Alpha) Using the
+ namespace field of dataSourceRef requires the
+ CrossNamespaceVolumeDataSource feature gate to
+ be enabled.'
+ properties:
+ apiGroup:
+ description: APIGroup is the group for the resource
+ being referenced. If APIGroup is not specified,
+ the specified Kind must be in the core API
+ group. For any other third-party types, APIGroup
+ is required.
+ type: string
+ kind:
+ description: Kind is the type of resource being
+ referenced
+ type: string
+ name:
+ description: Name is the name of resource being
+ referenced
+ type: string
+ namespace:
+ description: Namespace is the namespace of resource
+ being referenced Note that when a namespace
+ is specified, a gateway.networking.k8s.io/ReferenceGrant
+ object is required in the referent namespace
+ to allow that namespace's owner to accept
+ the reference. See the ReferenceGrant documentation
+ for details. (Alpha) This field requires the
+ CrossNamespaceVolumeDataSource feature gate
+ to be enabled.
+ type: string
+ required:
+ - kind
+ - name
+ type: object
+ resources:
+ description: 'resources represents the minimum resources
+ the volume should have. If RecoverVolumeExpansionFailure
+ feature is enabled users are allowed to specify
+ resource requirements that are lower than previous
+ value but must still be higher than capacity recorded
+ in the status field of the claim. More info: https://kubernetes.io/docs/concepts/storage/persistent-volumes#resources'
+ properties:
+ claims:
+ description: "Claims lists the names of resources,
+ defined in spec.resourceClaims, that are used
+ by this container. \n This is an alpha field
+ and requires enabling the DynamicResourceAllocation
+ feature gate. \n This field is immutable.
+ It can only be set for containers."
+ items:
+ description: ResourceClaim references one
+ entry in PodSpec.ResourceClaims.
+ properties:
+ name:
+ description: Name must match the name
+ of one entry in pod.spec.resourceClaims
+ of the Pod where this field is used.
+ It makes that resource available inside
+ a container.
+ type: string
+ required:
+ - name
+ type: object
+ type: array
+ x-kubernetes-list-map-keys:
+ - name
+ x-kubernetes-list-type: map
+ limits:
+ additionalProperties:
+ anyOf:
+ - type: integer
+ - type: string
+ pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$
+ x-kubernetes-int-or-string: true
+ description: 'Limits describes the maximum amount
+ of compute resources allowed. More info: https://kubernetes.io/docs/concepts/configuration/manage-resources-containers/'
+ type: object
+ requests:
+ additionalProperties:
+ anyOf:
+ - type: integer
+ - type: string
+ pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$
+ x-kubernetes-int-or-string: true
+ description: 'Requests describes the minimum
+ amount of compute resources required. If Requests
+ is omitted for a container, it defaults to
+ Limits if that is explicitly specified, otherwise
+ to an implementation-defined value. Requests
+ cannot exceed Limits. More info: https://kubernetes.io/docs/concepts/configuration/manage-resources-containers/'
+ type: object
+ type: object
+ selector:
+ description: selector is a label query over volumes
+ to consider for binding.
+ properties:
+ matchExpressions:
+ description: matchExpressions is a list of label
+ selector requirements. The requirements are
+ ANDed.
+ items:
+ description: A label selector requirement
+ is a selector that contains values, a key,
+ and an operator that relates the key and
+ values.
+ properties:
+ key:
+ description: key is the label key that
+ the selector applies to.
+ type: string
+ operator:
+ description: operator represents a key's
+ relationship to a set of values. Valid
+ operators are In, NotIn, Exists and
+ DoesNotExist.
+ type: string
+ values:
+ description: values is an array of string
+ values. If the operator is In or NotIn,
+ the values array must be non-empty.
+ If the operator is Exists or DoesNotExist,
+ the values array must be empty. This
+ array is replaced during a strategic
+ merge patch.
+ items:
+ type: string
+ type: array
+ required:
+ - key
+ - operator
+ type: object
+ type: array
+ matchLabels:
+ additionalProperties:
+ type: string
+ description: matchLabels is a map of {key,value}
+ pairs. A single {key,value} in the matchLabels
+ map is equivalent to an element of matchExpressions,
+ whose key field is "key", the operator is
+ "In", and the values array contains only "value".
+ The requirements are ANDed.
+ type: object
+ type: object
+ x-kubernetes-map-type: atomic
+ storageClassName:
+ description: 'storageClassName is the name of the
+ StorageClass required by the claim. More info:
+ https://kubernetes.io/docs/concepts/storage/persistent-volumes#class-1'
+ type: string
+ volumeMode:
+ description: volumeMode defines what type of volume
+ is required by the claim. Value of Filesystem
+ is implied when not included in claim spec.
+ type: string
+ volumeName:
+ description: volumeName is the binding reference
+ to the PersistentVolume backing this claim.
+ type: string
+ type: object
+ volumeSource:
+ description: volumeSource describes the volume will
+ be restored from the specified volume of the backup
+ targetVolumes. required if the backup uses volume
+ snapshot.
+ type: string
+ required:
+ - metadata
+ - volumeClaimSpec
+ type: object
+ x-kubernetes-validations:
+ - message: at least one exists for volumeSource and mountPath.
+ rule: self.volumeSource != '' || self.mountPath !=''
+ type: array
+ required:
+ - replicas
+ - templates
+ type: object
+ x-kubernetes-validations:
+ - message: forbidden to update spec.prepareDataConfig.volumeClaimsTemplate
+ rule: self == oldSelf
+ required:
+ - volumeClaimManagementPolicy
+ type: object
+ readyConfig:
+ description: configuration for the action of "postReady" phase.
+ properties:
+ connectCredential:
+ description: credential template used for creating a connection
+ credential
+ properties:
+ hostKey:
+ default: host
+ description: hostKey the map key of the host in the connection
+ credential secret
+ type: string
+ passwordKey:
+ default: password
+ description: passwordKey the map key of the password in the
+ connection credential secret
+ type: string
+ portKey:
+ default: port
+ description: portKey the map key of the port in the connection
+ credential secret
+ type: string
+ secretName:
+ description: the secret name
+ pattern: ^[a-z0-9]([a-z0-9\.\-]*[a-z0-9])?$
+ type: string
+ usernameKey:
+ default: username
+ description: usernameKey the map key of the user in the connection
+ credential secret
+ type: string
+ required:
+ - passwordKey
+ - secretName
+ - usernameKey
+ type: object
+ execAction:
+ description: configuration for exec action.
+ properties:
+ target:
+ description: execActionTarget defines the pods that need to
+ be executed for the exec action. will execute on all pods
+ that meet the conditions.
+ properties:
+ podSelector:
+ description: kubectl exec in all selected pods.
+ properties:
+ matchExpressions:
+ description: matchExpressions is a list of label selector
+ requirements. The requirements are ANDed.
+ items:
+ description: A label selector requirement is a selector
+ that contains values, a key, and an operator that
+ relates the key and values.
+ properties:
+ key:
+ description: key is the label key that the selector
+ applies to.
+ type: string
+ operator:
+ description: operator represents a key's relationship
+ to a set of values. Valid operators are In,
+ NotIn, Exists and DoesNotExist.
+ type: string
+ values:
+ description: values is an array of string values.
+ If the operator is In or NotIn, the values
+ array must be non-empty. If the operator is
+ Exists or DoesNotExist, the values array must
+ be empty. This array is replaced during a
+ strategic merge patch.
+ items:
+ type: string
+ type: array
+ required:
+ - key
+ - operator
+ type: object
+ type: array
+ matchLabels:
+ additionalProperties:
+ type: string
+ description: matchLabels is a map of {key,value} pairs.
+ A single {key,value} in the matchLabels map is equivalent
+ to an element of matchExpressions, whose key field
+ is "key", the operator is "In", and the values array
+ contains only "value". The requirements are ANDed.
+ type: object
+ type: object
+ x-kubernetes-map-type: atomic
+ required:
+ - podSelector
+ type: object
+ type: object
+ jobAction:
+ description: configuration for job action.
+ properties:
+ target:
+ description: jobActionTarget defines the pod that need to
+ be executed for the job action. will select a pod that meets
+ the conditions to execute.
+ properties:
+ podSelector:
+ description: select one of the pods which selected by
+ labels to build the job spec, such as mount required
+ volumes and inject built-in env of the selected pod.
+ properties:
+ matchExpressions:
+ description: matchExpressions is a list of label selector
+ requirements. The requirements are ANDed.
+ items:
+ description: A label selector requirement is a selector
+ that contains values, a key, and an operator that
+ relates the key and values.
+ properties:
+ key:
+ description: key is the label key that the selector
+ applies to.
+ type: string
+ operator:
+ description: operator represents a key's relationship
+ to a set of values. Valid operators are In,
+ NotIn, Exists and DoesNotExist.
+ type: string
+ values:
+ description: values is an array of string values.
+ If the operator is In or NotIn, the values
+ array must be non-empty. If the operator is
+ Exists or DoesNotExist, the values array must
+ be empty. This array is replaced during a
+ strategic merge patch.
+ items:
+ type: string
+ type: array
+ required:
+ - key
+ - operator
+ type: object
+ type: array
+ matchLabels:
+ additionalProperties:
+ type: string
+ description: matchLabels is a map of {key,value} pairs.
+ A single {key,value} in the matchLabels map is equivalent
+ to an element of matchExpressions, whose key field
+ is "key", the operator is "In", and the values array
+ contains only "value". The requirements are ANDed.
+ type: object
+ type: object
+ x-kubernetes-map-type: atomic
+ volumeMounts:
+ description: volumeMounts defines which volumes of the
+ selected pod need to be mounted on the restoring pod.
+ items:
+ description: VolumeMount describes a mounting of a Volume
+ within a container.
+ properties:
+ mountPath:
+ description: Path within the container at which
+ the volume should be mounted. Must not contain
+ ':'.
+ type: string
+ mountPropagation:
+ description: mountPropagation determines how mounts
+ are propagated from the host to container and
+ the other way around. When not set, MountPropagationNone
+ is used. This field is beta in 1.10.
+ type: string
+ name:
+ description: This must match the Name of a Volume.
+ type: string
+ readOnly:
+ description: Mounted read-only if true, read-write
+ otherwise (false or unspecified). Defaults to
+ false.
+ type: boolean
+ subPath:
+ description: Path within the volume from which the
+ container's volume should be mounted. Defaults
+ to "" (volume's root).
+ type: string
+ subPathExpr:
+ description: Expanded path within the volume from
+ which the container's volume should be mounted.
+ Behaves similarly to SubPath but environment variable
+ references $(VAR_NAME) are expanded using the
+ container's environment. Defaults to "" (volume's
+ root). SubPathExpr and SubPath are mutually exclusive.
+ type: string
+ required:
+ - mountPath
+ - name
+ type: object
+ type: array
+ required:
+ - podSelector
+ type: object
+ required:
+ - target
+ type: object
+ readinessProbe:
+ description: periodic probe of the service readiness. controller
+ will perform postReadyHooks of BackupScript.spec.restore after
+ the service readiness when readinessProbe is configured.
+ properties:
+ exec:
+ description: exec specifies the action to take.
+ properties:
+ command:
+ description: refer to container command.
+ items:
+ type: string
+ type: array
+ image:
+ description: refer to container image.
+ type: string
+ required:
+ - command
+ - image
+ type: object
+ initialDelaySeconds:
+ description: number of seconds after the container has started
+ before probe is initiated.
+ minimum: 0
+ type: integer
+ periodSeconds:
+ default: 5
+ description: how often (in seconds) to perform the probe.
+ defaults to 5 second, minimum value is 1.
+ minimum: 1
+ type: integer
+ timeoutSeconds:
+ default: 30
+ description: number of seconds after which the probe times
+ out. defaults to 30 second, minimum value is 1.
+ minimum: 1
+ type: integer
+ required:
+ - exec
+ type: object
+ type: object
+ x-kubernetes-validations:
+ - message: at least one exists for jobAction and execAction.
+ rule: has(self.jobAction) || has(self.execAction)
+ resources:
+ description: restore the specified resources of kubernetes.
+ properties:
+ included:
+ description: will restore the specified resources
+ items:
+ properties:
+ groupResource:
+ type: string
+ labelSelector:
+ description: select the specified resource for recovery
+ by label.
+ properties:
+ matchExpressions:
+ description: matchExpressions is a list of label selector
+ requirements. The requirements are ANDed.
+ items:
+ description: A label selector requirement is a selector
+ that contains values, a key, and an operator that
+ relates the key and values.
+ properties:
+ key:
+ description: key is the label key that the selector
+ applies to.
+ type: string
+ operator:
+ description: operator represents a key's relationship
+ to a set of values. Valid operators are In,
+ NotIn, Exists and DoesNotExist.
+ type: string
+ values:
+ description: values is an array of string values.
+ If the operator is In or NotIn, the values array
+ must be non-empty. If the operator is Exists
+ or DoesNotExist, the values array must be empty.
+ This array is replaced during a strategic merge
+ patch.
+ items:
+ type: string
+ type: array
+ required:
+ - key
+ - operator
+ type: object
+ type: array
+ matchLabels:
+ additionalProperties:
+ type: string
+ description: matchLabels is a map of {key,value} pairs.
+ A single {key,value} in the matchLabels map is equivalent
+ to an element of matchExpressions, whose key field
+ is "key", the operator is "In", and the values array
+ contains only "value". The requirements are ANDed.
+ type: object
+ type: object
+ x-kubernetes-map-type: atomic
+ required:
+ - groupResource
+ type: object
+ type: array
+ type: object
+ x-kubernetes-validations:
+ - message: forbidden to update spec.resources
+ rule: self == oldSelf
+ restoreTime:
+ description: restore according to a specified point in time.
+ pattern: ^\d{4}-\d{2}-\d{2}T\d{2}:\d{2}:\d{2}Z$
+ type: string
+ x-kubernetes-validations:
+ - message: forbidden to update spec.restoreTime
+ rule: self == oldSelf
+ serviceAccountName:
+ description: service account name which needs for recovery pod.
+ type: string
+ required:
+ - backup
+ type: object
+ status:
+ description: RestoreStatus defines the observed state of Restore
+ properties:
+ actions:
+ description: recorded all restore actions performed.
+ properties:
+ postReady:
+ description: record the actions for postReady phase.
+ items:
+ properties:
+ backupName:
+ description: which backup's restore action belongs to.
+ type: string
+ endTime:
+ description: endTime is the completion time for the restore
+ job.
+ format: date-time
+ type: string
+ message:
+ description: message is a human readable message indicating
+ details about the object condition.
+ type: string
+ name:
+ description: name describes the name of the recovery action
+ based on the current backup.
+ type: string
+ objectKey:
+ description: the execution object of the restore action.
+ type: string
+ startTime:
+ description: startTime is the start time for the restore
+ job.
+ format: date-time
+ type: string
+ status:
+ description: the status of this action.
+ enum:
+ - Processing
+ - Completed
+ - Failed
+ type: string
+ required:
+ - backupName
+ - name
+ - objectKey
+ type: object
+ type: array
+ prepareData:
+ description: record the actions for prepareData phase.
+ items:
+ properties:
+ backupName:
+ description: which backup's restore action belongs to.
+ type: string
+ endTime:
+ description: endTime is the completion time for the restore
+ job.
+ format: date-time
+ type: string
+ message:
+ description: message is a human readable message indicating
+ details about the object condition.
+ type: string
+ name:
+ description: name describes the name of the recovery action
+ based on the current backup.
+ type: string
+ objectKey:
+ description: the execution object of the restore action.
+ type: string
+ startTime:
+ description: startTime is the start time for the restore
+ job.
+ format: date-time
+ type: string
+ status:
+ description: the status of this action.
+ enum:
+ - Processing
+ - Completed
+ - Failed
+ type: string
+ required:
+ - backupName
+ - name
+ - objectKey
+ type: object
+ type: array
+ type: object
+ completionTimestamp:
+ description: Date/time when the restore finished being processed.
+ format: date-time
+ type: string
+ conditions:
+ description: describe current state of restore API Resource, like
+ warning.
+ items:
+ description: "Condition contains details for one aspect of the current
+ state of this API Resource. --- This struct is intended for direct
+ use as an array at the field path .status.conditions. For example,
+ \n type FooStatus struct{ // Represents the observations of a
+ foo's current state. // Known .status.conditions.type are: \"Available\",
+ \"Progressing\", and \"Degraded\" // +patchMergeKey=type // +patchStrategy=merge
+ // +listType=map // +listMapKey=type Conditions []metav1.Condition
+ `json:\"conditions,omitempty\" patchStrategy:\"merge\" patchMergeKey:\"type\"
+ protobuf:\"bytes,1,rep,name=conditions\"` \n // other fields }"
+ properties:
+ lastTransitionTime:
+ description: lastTransitionTime is the last time the condition
+ transitioned from one status to another. This should be when
+ the underlying condition changed. If that is not known, then
+ using the time when the API field changed is acceptable.
+ format: date-time
+ type: string
+ message:
+ description: message is a human readable message indicating
+ details about the transition. This may be an empty string.
+ maxLength: 32768
+ type: string
+ observedGeneration:
+ description: observedGeneration represents the .metadata.generation
+ that the condition was set based upon. For instance, if .metadata.generation
+ is currently 12, but the .status.conditions[x].observedGeneration
+ is 9, the condition is out of date with respect to the current
+ state of the instance.
+ format: int64
+ minimum: 0
+ type: integer
+ reason:
+ description: reason contains a programmatic identifier indicating
+ the reason for the condition's last transition. Producers
+ of specific condition types may define expected values and
+ meanings for this field, and whether the values are considered
+ a guaranteed API. The value should be a CamelCase string.
+ This field may not be empty.
+ maxLength: 1024
+ minLength: 1
+ pattern: ^[A-Za-z]([A-Za-z0-9_,:]*[A-Za-z0-9_])?$
+ type: string
+ status:
+ description: status of the condition, one of True, False, Unknown.
+ enum:
+ - "True"
+ - "False"
+ - Unknown
+ type: string
+ type:
+ description: type of condition in CamelCase or in foo.example.com/CamelCase.
+ --- Many .condition.type values are consistent across resources
+ like Available, but because arbitrary conditions can be useful
+ (see .node.status.conditions), the ability to deconflict is
+ important. The regex it matches is (dns1123SubdomainFmt/)?(qualifiedNameFmt)
+ maxLength: 316
+ pattern: ^([a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*/)?(([A-Za-z0-9][-A-Za-z0-9_.]*)?[A-Za-z0-9])$
+ type: string
+ required:
+ - lastTransitionTime
+ - message
+ - reason
+ - status
+ - type
+ type: object
+ type: array
+ duration:
+ description: The duration time of restore execution. When converted
+ to a string, the form is "1h2m0.5s".
+ type: string
+ phase:
+ description: RestorePhase The current phase. Valid values are Running,
+ Completed, Failed, Deleting.
+ enum:
+ - Running
+ - Completed
+ - Failed
+ - Deleting
+ type: string
+ startTimestamp:
+ description: Date/time when the restore started being processed.
+ format: date-time
+ type: string
+ type: object
+ type: object
+ served: true
+ storage: true
+ subresources:
+ status: {}
diff --git a/config/crd/bases/extensions.kubeblocks.io_addons.yaml b/config/crd/bases/extensions.kubeblocks.io_addons.yaml
index f170da7b08c..80f4c9f946b 100644
--- a/config/crd/bases/extensions.kubeblocks.io_addons.yaml
+++ b/config/crd/bases/extensions.kubeblocks.io_addons.yaml
@@ -2,8 +2,7 @@ apiVersion: apiextensions.k8s.io/v1
kind: CustomResourceDefinition
metadata:
annotations:
- controller-gen.kubebuilder.io/version: v0.9.0
- creationTimestamp: null
+ controller-gen.kubebuilder.io/version: v0.12.1
labels:
app.kubernetes.io/name: kubeblocks
name: addons.extensions.kubeblocks.io
@@ -172,12 +171,15 @@ spec:
properties:
key:
description: The selector key. Valid values are KubeVersion,
- KubeGitVersion. "KubeVersion" the semver expression
- of Kubernetes versions, i.e., v1.24. "KubeGitVersion"
- may contain distro. info., i.e., v1.24.4+eks.
+ KubeGitVersion and KubeProvider. "KubeVersion" the semver
+ expression of Kubernetes versions, i.e., v1.24. "KubeGitVersion"
+ may contain distro. info., i.e., v1.24.4+eks. "KubeProvider"
+ the Kubernetes provider, i.e., aws,gcp,azure,huaweiCloud,tencentCloud
+ etc.
enum:
- KubeGitVersion
- KubeVersion
+ - KubeProvider
type: string
operator:
description: "Represents a key's relationship to a set
@@ -562,12 +564,15 @@ spec:
properties:
key:
description: The selector key. Valid values are KubeVersion,
- KubeGitVersion. "KubeVersion" the semver expression of
- Kubernetes versions, i.e., v1.24. "KubeGitVersion" may
- contain distro. info., i.e., v1.24.4+eks.
+ KubeGitVersion and KubeProvider. "KubeVersion" the semver
+ expression of Kubernetes versions, i.e., v1.24. "KubeGitVersion"
+ may contain distro. info., i.e., v1.24.4+eks. "KubeProvider"
+ the Kubernetes provider, i.e., aws,gcp,azure,huaweiCloud,tencentCloud
+ etc.
enum:
- KubeGitVersion
- KubeVersion
+ - KubeProvider
type: string
operator:
description: "Represents a key's relationship to a set of
diff --git a/config/crd/bases/storage.kubeblocks.io_storageproviders.yaml b/config/crd/bases/storage.kubeblocks.io_storageproviders.yaml
index af55709f62e..5c2b75b65b3 100644
--- a/config/crd/bases/storage.kubeblocks.io_storageproviders.yaml
+++ b/config/crd/bases/storage.kubeblocks.io_storageproviders.yaml
@@ -2,8 +2,7 @@ apiVersion: apiextensions.k8s.io/v1
kind: CustomResourceDefinition
metadata:
annotations:
- controller-gen.kubebuilder.io/version: v0.9.0
- creationTimestamp: null
+ controller-gen.kubebuilder.io/version: v0.12.1
labels:
app.kubernetes.io/name: kubeblocks
name: storageproviders.storage.kubeblocks.io
@@ -18,7 +17,17 @@ spec:
singular: storageprovider
scope: Cluster
versions:
- - name: v1alpha1
+ - additionalPrinterColumns:
+ - jsonPath: .status.phase
+ name: STATUS
+ type: string
+ - jsonPath: .spec.csiDriverName
+ name: CSIDRIVER
+ type: string
+ - jsonPath: .metadata.creationTimestamp
+ name: AGE
+ type: date
+ name: v1alpha1
schema:
openAPIV3Schema:
description: StorageProvider is the Schema for the storageproviders API StorageProvider
@@ -48,6 +57,11 @@ spec:
by the CSI driver. The template will be rendered with the following
variables: - Parameters: a map of parameters defined in the ParametersSchema.'
type: string
+ datasafedConfigTemplate:
+ description: 'A Go template for rendering a config used by the datasafed
+ command. The template will be rendered with the following variables:
+ - Parameters: a map of parameters defined in the ParametersSchema.'
+ type: string
parametersSchema:
description: The schema describes the parameters required by this
StorageProvider, when rendering the templates.
@@ -64,6 +78,12 @@ spec:
type: object
x-kubernetes-preserve-unknown-fields: true
type: object
+ persistentVolumeClaimTemplate:
+ description: 'A Go template for rendering a PersistentVolumeClaim.
+ The template will be rendered with the following variables: - Parameters:
+ a map of parameters defined in the ParametersSchema. - GeneratedStorageClassName:
+ the name of the storage class generated with the StorageClassTemplate.'
+ type: string
storageClassTemplate:
description: 'A Go template for rendering a storage class which will
be used by the CSI driver. The template will be rendered with the
diff --git a/config/crd/bases/workloads.kubeblocks.io_replicatedstatemachines.yaml b/config/crd/bases/workloads.kubeblocks.io_replicatedstatemachines.yaml
index 2836ac341b7..6e84668baf6 100644
--- a/config/crd/bases/workloads.kubeblocks.io_replicatedstatemachines.yaml
+++ b/config/crd/bases/workloads.kubeblocks.io_replicatedstatemachines.yaml
@@ -2,8 +2,7 @@ apiVersion: apiextensions.k8s.io/v1
kind: CustomResourceDefinition
metadata:
annotations:
- controller-gen.kubebuilder.io/version: v0.9.0
- creationTimestamp: null
+ controller-gen.kubebuilder.io/version: v0.12.1
labels:
app.kubernetes.io/name: kubeblocks
name: replicatedstatemachines.workloads.kubeblocks.io
@@ -41,7 +40,7 @@ spec:
schema:
openAPIV3Schema:
description: ReplicatedStateMachine is the Schema for the replicatedstatemachines
- API
+ API.
properties:
apiVersion:
description: 'APIVersion defines the versioned schema of this representation
@@ -287,11 +286,10 @@ spec:
supports specifying the loadBalancerIP when a load balancer
is created. This field will be ignored if the cloud-provider
does not support the feature. Deprecated: This field was
- under-specified and its meaning varies across implementations,
- and it cannot support dual-stack. As of Kubernetes v1.24,
- users are encouraged to use implementation-specific annotations
- when available. This field may be removed in a future
- API version.'
+ under-specified and its meaning varies across implementations.
+ Using it is non-portable and it may not support dual-stack.
+ Users are encouraged to use implementation-specific annotations
+ when available.'
type: string
loadBalancerSourceRanges:
description: 'If specified and supported by the platform,
@@ -310,12 +308,21 @@ spec:
port.
properties:
appProtocol:
- description: The application protocol for this port.
+ description: "The application protocol for this port.
+ This is used as a hint for implementations to offer
+ richer behavior for protocols that they understand.
This field follows standard Kubernetes label syntax.
- Un-prefixed names are reserved for IANA standard
- service names (as per RFC-6335 and https://www.iana.org/assignments/service-names).
- Non-standard protocols should use prefixed names
- such as mycompany.com/my-custom-protocol.
+ Valid values are either: \n * Un-prefixed protocol
+ names - reserved for IANA standard service names
+ (as per RFC-6335 and https://www.iana.org/assignments/service-names).
+ \n * Kubernetes-defined prefixed names: * 'kubernetes.io/h2c'
+ - HTTP/2 over cleartext as described in https://www.rfc-editor.org/rfc/rfc7540
+ * 'kubernetes.io/ws' - WebSocket over cleartext
+ as described in https://www.rfc-editor.org/rfc/rfc6455
+ * 'kubernetes.io/wss' - WebSocket over TLS as described
+ in https://www.rfc-editor.org/rfc/rfc6455 \n * Other
+ protocols should use implementation-defined prefixed
+ names such as mycompany.com/my-custom-protocol."
type: string
name:
description: The name of this port within the service.
@@ -630,6 +637,7 @@ spec:
required:
- key
type: object
+ x-kubernetes-map-type: atomic
fieldRef:
description: 'Selects a field of the pod: supports metadata.name,
metadata.namespace, `metadata.labels['''']`, `metadata.annotations['''']`,
@@ -647,6 +655,7 @@ spec:
required:
- fieldPath
type: object
+ x-kubernetes-map-type: atomic
resourceFieldRef:
description: 'Selects a resource of the container: only
resources limits and requests (limits.cpu, limits.memory,
@@ -671,6 +680,7 @@ spec:
required:
- resource
type: object
+ x-kubernetes-map-type: atomic
secretKeyRef:
description: Selects a key of a secret in the pod's namespace
properties:
@@ -690,6 +700,7 @@ spec:
required:
- key
type: object
+ x-kubernetes-map-type: atomic
type: object
type: object
username:
@@ -728,6 +739,7 @@ spec:
required:
- key
type: object
+ x-kubernetes-map-type: atomic
fieldRef:
description: 'Selects a field of the pod: supports metadata.name,
metadata.namespace, `metadata.labels['''']`, `metadata.annotations['''']`,
@@ -745,6 +757,7 @@ spec:
required:
- fieldPath
type: object
+ x-kubernetes-map-type: atomic
resourceFieldRef:
description: 'Selects a resource of the container: only
resources limits and requests (limits.cpu, limits.memory,
@@ -769,6 +782,7 @@ spec:
required:
- resource
type: object
+ x-kubernetes-map-type: atomic
secretKeyRef:
description: Selects a key of a secret in the pod's namespace
properties:
@@ -788,6 +802,7 @@ spec:
required:
- key
type: object
+ x-kubernetes-map-type: atomic
type: object
type: object
required:
@@ -970,6 +985,15 @@ spec:
- command
type: object
type: array
+ roleUpdateMechanism:
+ default: None
+ description: RoleUpdateMechanism specifies the way how pod role
+ label being updated.
+ enum:
+ - ReadinessProbeEventUpdate
+ - DirectAPIServerEventUpdate
+ - None
+ type: string
successThreshold:
default: 1
description: Minimum consecutive successes for the probe to be
@@ -1063,6 +1087,7 @@ spec:
are ANDed.
type: object
type: object
+ x-kubernetes-map-type: atomic
service:
description: service defines the behavior of a service spec. provides
read-write service https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#spec-and-status
@@ -1284,11 +1309,10 @@ spec:
supports specifying the loadBalancerIP when a load balancer
is created. This field will be ignored if the cloud-provider
does not support the feature. Deprecated: This field was
- under-specified and its meaning varies across implementations,
- and it cannot support dual-stack. As of Kubernetes v1.24,
- users are encouraged to use implementation-specific annotations
- when available. This field may be removed in a future API
- version.'
+ under-specified and its meaning varies across implementations.
+ Using it is non-portable and it may not support dual-stack.
+ Users are encouraged to use implementation-specific annotations
+ when available.'
type: string
loadBalancerSourceRanges:
description: 'If specified and supported by the platform,
@@ -1307,12 +1331,21 @@ spec:
port.
properties:
appProtocol:
- description: The application protocol for this port.
+ description: "The application protocol for this port.
+ This is used as a hint for implementations to offer
+ richer behavior for protocols that they understand.
This field follows standard Kubernetes label syntax.
- Un-prefixed names are reserved for IANA standard service
- names (as per RFC-6335 and https://www.iana.org/assignments/service-names).
- Non-standard protocols should use prefixed names such
- as mycompany.com/my-custom-protocol.
+ Valid values are either: \n * Un-prefixed protocol
+ names - reserved for IANA standard service names (as
+ per RFC-6335 and https://www.iana.org/assignments/service-names).
+ \n * Kubernetes-defined prefixed names: * 'kubernetes.io/h2c'
+ - HTTP/2 over cleartext as described in https://www.rfc-editor.org/rfc/rfc7540
+ * 'kubernetes.io/ws' - WebSocket over cleartext as
+ described in https://www.rfc-editor.org/rfc/rfc6455
+ * 'kubernetes.io/wss' - WebSocket over TLS as described
+ in https://www.rfc-editor.org/rfc/rfc6455 \n * Other
+ protocols should use implementation-defined prefixed
+ names such as mycompany.com/my-custom-protocol."
type: string
name:
description: The name of this port within the service.
@@ -1729,6 +1762,7 @@ spec:
type: object
type: array
type: object
+ x-kubernetes-map-type: atomic
weight:
description: Weight associated with matching
the corresponding nodeSelectorTerm, in the
@@ -1835,10 +1869,12 @@ spec:
type: object
type: array
type: object
+ x-kubernetes-map-type: atomic
type: array
required:
- nodeSelectorTerms
type: object
+ x-kubernetes-map-type: atomic
type: object
podAffinity:
description: Describes pod affinity scheduling rules (e.g.
@@ -1923,6 +1959,7 @@ spec:
ANDed.
type: object
type: object
+ x-kubernetes-map-type: atomic
namespaceSelector:
description: A label query over the set
of namespaces that the term applies to.
@@ -1984,6 +2021,7 @@ spec:
ANDed.
type: object
type: object
+ x-kubernetes-map-type: atomic
namespaces:
description: namespaces specifies a static
list of namespace names that the term
@@ -2093,6 +2131,7 @@ spec:
only "value". The requirements are ANDed.
type: object
type: object
+ x-kubernetes-map-type: atomic
namespaceSelector:
description: A label query over the set of namespaces
that the term applies to. The term is applied
@@ -2150,6 +2189,7 @@ spec:
only "value". The requirements are ANDed.
type: object
type: object
+ x-kubernetes-map-type: atomic
namespaces:
description: namespaces specifies a static list
of namespace names that the term applies to.
@@ -2259,6 +2299,7 @@ spec:
ANDed.
type: object
type: object
+ x-kubernetes-map-type: atomic
namespaceSelector:
description: A label query over the set
of namespaces that the term applies to.
@@ -2320,6 +2361,7 @@ spec:
ANDed.
type: object
type: object
+ x-kubernetes-map-type: atomic
namespaces:
description: namespaces specifies a static
list of namespace names that the term
@@ -2429,6 +2471,7 @@ spec:
only "value". The requirements are ANDed.
type: object
type: object
+ x-kubernetes-map-type: atomic
namespaceSelector:
description: A label query over the set of namespaces
that the term applies to. The term is applied
@@ -2486,6 +2529,7 @@ spec:
only "value". The requirements are ANDed.
type: object
type: object
+ x-kubernetes-map-type: atomic
namespaces:
description: namespaces specifies a static list
of namespace names that the term applies to.
@@ -2602,6 +2646,7 @@ spec:
required:
- key
type: object
+ x-kubernetes-map-type: atomic
fieldRef:
description: 'Selects a field of the pod:
supports metadata.name, metadata.namespace,
@@ -2621,6 +2666,7 @@ spec:
required:
- fieldPath
type: object
+ x-kubernetes-map-type: atomic
resourceFieldRef:
description: 'Selects a resource of the container:
only resources limits and requests (limits.cpu,
@@ -2647,6 +2693,7 @@ spec:
required:
- resource
type: object
+ x-kubernetes-map-type: atomic
secretKeyRef:
description: Selects a key of a secret in
the pod's namespace
@@ -2669,6 +2716,7 @@ spec:
required:
- key
type: object
+ x-kubernetes-map-type: atomic
type: object
required:
- name
@@ -2701,6 +2749,7 @@ spec:
must be defined
type: boolean
type: object
+ x-kubernetes-map-type: atomic
prefix:
description: An optional identifier to prepend
to each key in the ConfigMap. Must be a C_IDENTIFIER.
@@ -2719,6 +2768,7 @@ spec:
be defined
type: boolean
type: object
+ x-kubernetes-map-type: atomic
type: object
type: array
image:
@@ -2781,7 +2831,10 @@ spec:
header to be used in HTTP probes
properties:
name:
- description: The header field name
+ description: The header field name.
+ This will be canonicalized upon
+ output, so case-variant names will
+ be understood as the same header.
type: string
value:
description: The header field value
@@ -2885,7 +2938,10 @@ spec:
header to be used in HTTP probes
properties:
name:
- description: The header field name
+ description: The header field name.
+ This will be canonicalized upon
+ output, so case-variant names will
+ be understood as the same header.
type: string
value:
description: The header field value
@@ -2970,8 +3026,7 @@ spec:
type: integer
grpc:
description: GRPC specifies an action involving
- a GRPC port. This is a beta field and requires
- enabling GRPCContainerProbe feature gate.
+ a GRPC port.
properties:
port:
description: Port number of the gRPC service.
@@ -3005,7 +3060,10 @@ spec:
header to be used in HTTP probes
properties:
name:
- description: The header field name
+ description: The header field name. This
+ will be canonicalized upon output, so
+ case-variant names will be understood
+ as the same header.
type: string
value:
description: The header field value
@@ -3182,8 +3240,7 @@ spec:
type: integer
grpc:
description: GRPC specifies an action involving
- a GRPC port. This is a beta field and requires
- enabling GRPCContainerProbe feature gate.
+ a GRPC port.
properties:
port:
description: Port number of the gRPC service.
@@ -3217,7 +3274,10 @@ spec:
header to be used in HTTP probes
properties:
name:
- description: The header field name
+ description: The header field name. This
+ will be canonicalized upon output, so
+ case-variant names will be understood
+ as the same header.
type: string
value:
description: The header field value
@@ -3309,6 +3369,28 @@ spec:
format: int32
type: integer
type: object
+ resizePolicy:
+ description: Resources resize policy for the container.
+ items:
+ description: ContainerResizePolicy represents resource
+ resize policy for the container.
+ properties:
+ resourceName:
+ description: 'Name of the resource to which this
+ resource resize policy applies. Supported values:
+ cpu, memory.'
+ type: string
+ restartPolicy:
+ description: Restart policy to apply when specified
+ resource is resized. If not specified, it defaults
+ to NotRequired.
+ type: string
+ required:
+ - resourceName
+ - restartPolicy
+ type: object
+ type: array
+ x-kubernetes-list-type: atomic
resources:
description: 'Compute Resources required by this container.
Cannot be updated. More info: https://kubernetes.io/docs/concepts/configuration/manage-resources-containers/'
@@ -3358,9 +3440,32 @@ spec:
of compute resources required. If Requests is
omitted for a container, it defaults to Limits
if that is explicitly specified, otherwise to
- an implementation-defined value. More info: https://kubernetes.io/docs/concepts/configuration/manage-resources-containers/'
+ an implementation-defined value. Requests cannot
+ exceed Limits. More info: https://kubernetes.io/docs/concepts/configuration/manage-resources-containers/'
type: object
type: object
+ restartPolicy:
+ description: 'RestartPolicy defines the restart behavior
+ of individual containers in a pod. This field may
+ only be set for init containers, and the only allowed
+ value is "Always". For non-init containers or when
+ this field is not specified, the restart behavior
+ is defined by the Pod''s restart policy and the container
+ type. Setting the RestartPolicy as "Always" for the
+ init container will have the following effect: this
+ init container will be continually restarted on exit
+ until all regular containers have terminated. Once
+ all regular containers have completed, all init containers
+ with restartPolicy "Always" will be shut down. This
+ lifecycle differs from normal init containers and
+ is often referred to as a "sidecar" container. Although
+ this init container still starts in the init container
+ sequence, it does not wait for the container to complete
+ before proceeding to the next init container. Instead,
+ the next init container starts immediately after this
+ init container is started, or after any startupProbe
+ has successfully completed.'
+ type: string
securityContext:
description: 'SecurityContext defines the security options
the container should be run with. If set, the fields
@@ -3490,7 +3595,8 @@ spec:
The profile must be preconfigured on the node
to work. Must be a descending path, relative
to the kubelet's configured seccomp profile
- location. Must only be set if type is "Localhost".
+ location. Must be set if type is "Localhost".
+ Must NOT be set for any other type.
type: string
type:
description: "type indicates which kind of seccomp
@@ -3526,14 +3632,10 @@ spec:
hostProcess:
description: HostProcess determines if a container
should be run as a 'Host Process' container.
- This field is alpha-level and will only be
- honored by components that enable the WindowsHostProcessContainers
- feature flag. Setting this field without the
- feature flag will result in errors when validating
- the Pod. All of a Pod's containers must have
- the same effective HostProcess value (it is
- not allowed to have a mix of HostProcess containers
- and non-HostProcess containers). In addition,
+ All of a Pod's containers must have the same
+ effective HostProcess value (it is not allowed
+ to have a mix of HostProcess containers and
+ non-HostProcess containers). In addition,
if HostProcess is true then HostNetwork must
also be set to true.
type: boolean
@@ -3584,8 +3686,7 @@ spec:
type: integer
grpc:
description: GRPC specifies an action involving
- a GRPC port. This is a beta field and requires
- enabling GRPCContainerProbe feature gate.
+ a GRPC port.
properties:
port:
description: Port number of the gRPC service.
@@ -3619,7 +3720,10 @@ spec:
header to be used in HTTP probes
properties:
name:
- description: The header field name
+ description: The header field name. This
+ will be canonicalized upon output, so
+ case-variant names will be understood
+ as the same header.
type: string
value:
description: The header field value
@@ -3981,6 +4085,7 @@ spec:
required:
- key
type: object
+ x-kubernetes-map-type: atomic
fieldRef:
description: 'Selects a field of the pod:
supports metadata.name, metadata.namespace,
@@ -4000,6 +4105,7 @@ spec:
required:
- fieldPath
type: object
+ x-kubernetes-map-type: atomic
resourceFieldRef:
description: 'Selects a resource of the container:
only resources limits and requests (limits.cpu,
@@ -4026,6 +4132,7 @@ spec:
required:
- resource
type: object
+ x-kubernetes-map-type: atomic
secretKeyRef:
description: Selects a key of a secret in
the pod's namespace
@@ -4048,6 +4155,7 @@ spec:
required:
- key
type: object
+ x-kubernetes-map-type: atomic
type: object
required:
- name
@@ -4080,6 +4188,7 @@ spec:
must be defined
type: boolean
type: object
+ x-kubernetes-map-type: atomic
prefix:
description: An optional identifier to prepend
to each key in the ConfigMap. Must be a C_IDENTIFIER.
@@ -4098,6 +4207,7 @@ spec:
be defined
type: boolean
type: object
+ x-kubernetes-map-type: atomic
type: object
type: array
image:
@@ -4156,7 +4266,10 @@ spec:
header to be used in HTTP probes
properties:
name:
- description: The header field name
+ description: The header field name.
+ This will be canonicalized upon
+ output, so case-variant names will
+ be understood as the same header.
type: string
value:
description: The header field value
@@ -4260,7 +4373,10 @@ spec:
header to be used in HTTP probes
properties:
name:
- description: The header field name
+ description: The header field name.
+ This will be canonicalized upon
+ output, so case-variant names will
+ be understood as the same header.
type: string
value:
description: The header field value
@@ -4343,8 +4459,7 @@ spec:
type: integer
grpc:
description: GRPC specifies an action involving
- a GRPC port. This is a beta field and requires
- enabling GRPCContainerProbe feature gate.
+ a GRPC port.
properties:
port:
description: Port number of the gRPC service.
@@ -4378,7 +4493,10 @@ spec:
header to be used in HTTP probes
properties:
name:
- description: The header field name
+ description: The header field name. This
+ will be canonicalized upon output, so
+ case-variant names will be understood
+ as the same header.
type: string
value:
description: The header field value
@@ -4546,8 +4664,7 @@ spec:
type: integer
grpc:
description: GRPC specifies an action involving
- a GRPC port. This is a beta field and requires
- enabling GRPCContainerProbe feature gate.
+ a GRPC port.
properties:
port:
description: Port number of the gRPC service.
@@ -4581,7 +4698,10 @@ spec:
header to be used in HTTP probes
properties:
name:
- description: The header field name
+ description: The header field name. This
+ will be canonicalized upon output, so
+ case-variant names will be understood
+ as the same header.
type: string
value:
description: The header field value
@@ -4673,6 +4793,28 @@ spec:
format: int32
type: integer
type: object
+ resizePolicy:
+ description: Resources resize policy for the container.
+ items:
+ description: ContainerResizePolicy represents resource
+ resize policy for the container.
+ properties:
+ resourceName:
+ description: 'Name of the resource to which this
+ resource resize policy applies. Supported values:
+ cpu, memory.'
+ type: string
+ restartPolicy:
+ description: Restart policy to apply when specified
+ resource is resized. If not specified, it defaults
+ to NotRequired.
+ type: string
+ required:
+ - resourceName
+ - restartPolicy
+ type: object
+ type: array
+ x-kubernetes-list-type: atomic
resources:
description: Resources are not allowed for ephemeral
containers. Ephemeral containers use spare resources
@@ -4723,9 +4865,16 @@ spec:
of compute resources required. If Requests is
omitted for a container, it defaults to Limits
if that is explicitly specified, otherwise to
- an implementation-defined value. More info: https://kubernetes.io/docs/concepts/configuration/manage-resources-containers/'
+ an implementation-defined value. Requests cannot
+ exceed Limits. More info: https://kubernetes.io/docs/concepts/configuration/manage-resources-containers/'
type: object
type: object
+ restartPolicy:
+ description: Restart policy for the container to manage
+ the restart behavior of each container within a pod.
+ This may only be set for init containers. You cannot
+ set this field on ephemeral containers.
+ type: string
securityContext:
description: 'Optional: SecurityContext defines the
security options the ephemeral container should be
@@ -4855,7 +5004,8 @@ spec:
The profile must be preconfigured on the node
to work. Must be a descending path, relative
to the kubelet's configured seccomp profile
- location. Must only be set if type is "Localhost".
+ location. Must be set if type is "Localhost".
+ Must NOT be set for any other type.
type: string
type:
description: "type indicates which kind of seccomp
@@ -4891,14 +5041,10 @@ spec:
hostProcess:
description: HostProcess determines if a container
should be run as a 'Host Process' container.
- This field is alpha-level and will only be
- honored by components that enable the WindowsHostProcessContainers
- feature flag. Setting this field without the
- feature flag will result in errors when validating
- the Pod. All of a Pod's containers must have
- the same effective HostProcess value (it is
- not allowed to have a mix of HostProcess containers
- and non-HostProcess containers). In addition,
+ All of a Pod's containers must have the same
+ effective HostProcess value (it is not allowed
+ to have a mix of HostProcess containers and
+ non-HostProcess containers). In addition,
if HostProcess is true then HostNetwork must
also be set to true.
type: boolean
@@ -4941,8 +5087,7 @@ spec:
type: integer
grpc:
description: GRPC specifies an action involving
- a GRPC port. This is a beta field and requires
- enabling GRPCContainerProbe feature gate.
+ a GRPC port.
properties:
port:
description: Port number of the gRPC service.
@@ -4976,7 +5121,10 @@ spec:
header to be used in HTTP probes
properties:
name:
- description: The header field name
+ description: The header field name. This
+ will be canonicalized upon output, so
+ case-variant names will be understood
+ as the same header.
type: string
value:
description: The header field value
@@ -5267,6 +5415,7 @@ spec:
TODO: Add other useful fields. apiVersion, kind, uid?'
type: string
type: object
+ x-kubernetes-map-type: atomic
type: array
initContainers:
description: 'List of initialization containers belonging
@@ -5364,6 +5513,7 @@ spec:
required:
- key
type: object
+ x-kubernetes-map-type: atomic
fieldRef:
description: 'Selects a field of the pod:
supports metadata.name, metadata.namespace,
@@ -5383,6 +5533,7 @@ spec:
required:
- fieldPath
type: object
+ x-kubernetes-map-type: atomic
resourceFieldRef:
description: 'Selects a resource of the container:
only resources limits and requests (limits.cpu,
@@ -5409,6 +5560,7 @@ spec:
required:
- resource
type: object
+ x-kubernetes-map-type: atomic
secretKeyRef:
description: Selects a key of a secret in
the pod's namespace
@@ -5431,6 +5583,7 @@ spec:
required:
- key
type: object
+ x-kubernetes-map-type: atomic
type: object
required:
- name
@@ -5463,6 +5616,7 @@ spec:
must be defined
type: boolean
type: object
+ x-kubernetes-map-type: atomic
prefix:
description: An optional identifier to prepend
to each key in the ConfigMap. Must be a C_IDENTIFIER.
@@ -5481,6 +5635,7 @@ spec:
be defined
type: boolean
type: object
+ x-kubernetes-map-type: atomic
type: object
type: array
image:
@@ -5543,7 +5698,10 @@ spec:
header to be used in HTTP probes
properties:
name:
- description: The header field name
+ description: The header field name.
+ This will be canonicalized upon
+ output, so case-variant names will
+ be understood as the same header.
type: string
value:
description: The header field value
@@ -5647,7 +5805,10 @@ spec:
header to be used in HTTP probes
properties:
name:
- description: The header field name
+ description: The header field name.
+ This will be canonicalized upon
+ output, so case-variant names will
+ be understood as the same header.
type: string
value:
description: The header field value
@@ -5732,8 +5893,7 @@ spec:
type: integer
grpc:
description: GRPC specifies an action involving
- a GRPC port. This is a beta field and requires
- enabling GRPCContainerProbe feature gate.
+ a GRPC port.
properties:
port:
description: Port number of the gRPC service.
@@ -5767,7 +5927,10 @@ spec:
header to be used in HTTP probes
properties:
name:
- description: The header field name
+ description: The header field name. This
+ will be canonicalized upon output, so
+ case-variant names will be understood
+ as the same header.
type: string
value:
description: The header field value
@@ -5944,8 +6107,7 @@ spec:
type: integer
grpc:
description: GRPC specifies an action involving
- a GRPC port. This is a beta field and requires
- enabling GRPCContainerProbe feature gate.
+ a GRPC port.
properties:
port:
description: Port number of the gRPC service.
@@ -5979,7 +6141,10 @@ spec:
header to be used in HTTP probes
properties:
name:
- description: The header field name
+ description: The header field name. This
+ will be canonicalized upon output, so
+ case-variant names will be understood
+ as the same header.
type: string
value:
description: The header field value
@@ -6071,6 +6236,28 @@ spec:
format: int32
type: integer
type: object
+ resizePolicy:
+ description: Resources resize policy for the container.
+ items:
+ description: ContainerResizePolicy represents resource
+ resize policy for the container.
+ properties:
+ resourceName:
+ description: 'Name of the resource to which this
+ resource resize policy applies. Supported values:
+ cpu, memory.'
+ type: string
+ restartPolicy:
+ description: Restart policy to apply when specified
+ resource is resized. If not specified, it defaults
+ to NotRequired.
+ type: string
+ required:
+ - resourceName
+ - restartPolicy
+ type: object
+ type: array
+ x-kubernetes-list-type: atomic
resources:
description: 'Compute Resources required by this container.
Cannot be updated. More info: https://kubernetes.io/docs/concepts/configuration/manage-resources-containers/'
@@ -6120,9 +6307,32 @@ spec:
of compute resources required. If Requests is
omitted for a container, it defaults to Limits
if that is explicitly specified, otherwise to
- an implementation-defined value. More info: https://kubernetes.io/docs/concepts/configuration/manage-resources-containers/'
+ an implementation-defined value. Requests cannot
+ exceed Limits. More info: https://kubernetes.io/docs/concepts/configuration/manage-resources-containers/'
type: object
type: object
+ restartPolicy:
+ description: 'RestartPolicy defines the restart behavior
+ of individual containers in a pod. This field may
+ only be set for init containers, and the only allowed
+ value is "Always". For non-init containers or when
+ this field is not specified, the restart behavior
+ is defined by the Pod''s restart policy and the container
+ type. Setting the RestartPolicy as "Always" for the
+ init container will have the following effect: this
+ init container will be continually restarted on exit
+ until all regular containers have terminated. Once
+ all regular containers have completed, all init containers
+ with restartPolicy "Always" will be shut down. This
+ lifecycle differs from normal init containers and
+ is often referred to as a "sidecar" container. Although
+ this init container still starts in the init container
+ sequence, it does not wait for the container to complete
+ before proceeding to the next init container. Instead,
+ the next init container starts immediately after this
+ init container is started, or after any startupProbe
+ has successfully completed.'
+ type: string
securityContext:
description: 'SecurityContext defines the security options
the container should be run with. If set, the fields
@@ -6252,7 +6462,8 @@ spec:
The profile must be preconfigured on the node
to work. Must be a descending path, relative
to the kubelet's configured seccomp profile
- location. Must only be set if type is "Localhost".
+ location. Must be set if type is "Localhost".
+ Must NOT be set for any other type.
type: string
type:
description: "type indicates which kind of seccomp
@@ -6288,14 +6499,10 @@ spec:
hostProcess:
description: HostProcess determines if a container
should be run as a 'Host Process' container.
- This field is alpha-level and will only be
- honored by components that enable the WindowsHostProcessContainers
- feature flag. Setting this field without the
- feature flag will result in errors when validating
- the Pod. All of a Pod's containers must have
- the same effective HostProcess value (it is
- not allowed to have a mix of HostProcess containers
- and non-HostProcess containers). In addition,
+ All of a Pod's containers must have the same
+ effective HostProcess value (it is not allowed
+ to have a mix of HostProcess containers and
+ non-HostProcess containers). In addition,
if HostProcess is true then HostNetwork must
also be set to true.
type: boolean
@@ -6346,8 +6553,7 @@ spec:
type: integer
grpc:
description: GRPC specifies an action involving
- a GRPC port. This is a beta field and requires
- enabling GRPCContainerProbe feature gate.
+ a GRPC port.
properties:
port:
description: Port number of the gRPC service.
@@ -6381,7 +6587,10 @@ spec:
header to be used in HTTP probes
properties:
name:
- description: The header field name
+ description: The header field name. This
+ will be canonicalized upon output, so
+ case-variant names will be understood
+ as the same header.
type: string
value:
description: The header field value
@@ -6729,19 +6938,14 @@ spec:
namespace as this pod. \n The template will be
used to create a new ResourceClaim, which will
be bound to this pod. When this pod is deleted,
- the ResourceClaim will also be deleted. The name
- of the ResourceClaim will be -, where is the PodResourceClaim.Name.
- Pod validation will reject the pod if the concatenated
- name is not valid for a ResourceClaim (e.g. too
- long). \n An existing ResourceClaim with that
- name that is not owned by the pod will not be
- used for the pod to avoid using an unrelated resource
- by mistake. Scheduling and pod startup are then
- blocked until the unrelated ResourceClaim is removed.
- \n This field is immutable and no changes will
- be made to the corresponding ResourceClaim by
- the control plane after creating the ResourceClaim."
+ the ResourceClaim will also be deleted. The pod
+ name and resource name, along with a generated
+ component, will be used to form a unique name
+ for the ResourceClaim, which will be recorded
+ in pod.status.resourceClaimStatuses. \n This field
+ is immutable and no changes will be made to the
+ corresponding ResourceClaim by the control plane
+ after creating the ResourceClaim."
type: string
type: object
required:
@@ -6753,8 +6957,9 @@ spec:
x-kubernetes-list-type: map
restartPolicy:
description: 'Restart policy for all containers within the
- pod. One of Always, OnFailure, Never. Default to Always.
- More info: https://kubernetes.io/docs/concepts/workloads/pods/pod-lifecycle/#restart-policy'
+ pod. One of Always, OnFailure, Never. In some contexts,
+ only a subset of those values may be permitted. Default
+ to Always. More info: https://kubernetes.io/docs/concepts/workloads/pods/pod-lifecycle/#restart-policy'
type: string
runtimeClassName:
description: 'RuntimeClassName refers to a RuntimeClass object
@@ -6772,10 +6977,12 @@ spec:
type: string
schedulingGates:
description: "SchedulingGates is an opaque list of values
- that if specified will block scheduling the pod. More info:
- \ https://git.k8s.io/enhancements/keps/sig-scheduling/3521-pod-scheduling-readiness.
- \n This is an alpha-level feature enabled by PodSchedulingReadiness
- feature gate."
+ that if specified will block scheduling the pod. If schedulingGates
+ is not empty, the pod will stay in the SchedulingGated state
+ and the scheduler will not attempt to schedule the pod.
+ \n SchedulingGates can only be set at pod creation time,
+ and be removed only afterwards. \n This is a beta feature
+ enabled by the PodSchedulingReadiness feature gate."
items:
description: PodSchedulingGate is associated to a Pod to
guard its scheduling.
@@ -6885,8 +7092,9 @@ spec:
defined in a file on the node should be used. The
profile must be preconfigured on the node to work.
Must be a descending path, relative to the kubelet's
- configured seccomp profile location. Must only be
- set if type is "Localhost".
+ configured seccomp profile location. Must be set
+ if type is "Localhost". Must NOT be set for any
+ other type.
type: string
type:
description: "type indicates which kind of seccomp
@@ -6954,15 +7162,11 @@ spec:
type: string
hostProcess:
description: HostProcess determines if a container
- should be run as a 'Host Process' container. This
- field is alpha-level and will only be honored by
- components that enable the WindowsHostProcessContainers
- feature flag. Setting this field without the feature
- flag will result in errors when validating the Pod.
- All of a Pod's containers must have the same effective
+ should be run as a 'Host Process' container. All
+ of a Pod's containers must have the same effective
HostProcess value (it is not allowed to have a mix
- of HostProcess containers and non-HostProcess containers). In
- addition, if HostProcess is true then HostNetwork
+ of HostProcess containers and non-HostProcess containers).
+ In addition, if HostProcess is true then HostNetwork
must also be set to true.
type: boolean
runAsUserName:
@@ -7122,16 +7326,21 @@ spec:
requirements are ANDed.
type: object
type: object
+ x-kubernetes-map-type: atomic
matchLabelKeys:
- description: MatchLabelKeys is a set of pod label keys
+ description: "MatchLabelKeys is a set of pod label keys
to select the pods over which spreading will be calculated.
The keys are used to lookup values from the incoming
pod labels, those key-value labels are ANDed with
labelSelector to select the group of existing pods
over which spreading will be calculated for the incoming
- pod. Keys that don't exist in the incoming pod labels
- will be ignored. A null or empty list means only match
- against labelSelector.
+ pod. The same key is forbidden to exist in both MatchLabelKeys
+ and LabelSelector. MatchLabelKeys cannot be set when
+ LabelSelector isn't set. Keys that don't exist in
+ the incoming pod labels will be ignored. A null or
+ empty list means only match against labelSelector.
+ \n This is a beta field and requires the MatchLabelKeysInPodTopologySpread
+ feature gate to be enabled (enabled by default)."
items:
type: string
type: array
@@ -7391,6 +7600,7 @@ spec:
kind, uid?'
type: string
type: object
+ x-kubernetes-map-type: atomic
user:
description: 'user is optional: User is the rados
user name, default is admin More info: https://examples.k8s.io/volumes/cephfs/README.md#how-to-use-it'
@@ -7426,6 +7636,7 @@ spec:
kind, uid?'
type: string
type: object
+ x-kubernetes-map-type: atomic
volumeID:
description: 'volumeID used to identify the volume
in cinder. More info: https://examples.k8s.io/mysql-cinder-pd/README.md'
@@ -7505,6 +7716,7 @@ spec:
or its keys must be defined
type: boolean
type: object
+ x-kubernetes-map-type: atomic
csi:
description: csi (Container Storage Interface) represents
ephemeral storage that is handled by certain external
@@ -7538,6 +7750,7 @@ spec:
kind, uid?'
type: string
type: object
+ x-kubernetes-map-type: atomic
readOnly:
description: readOnly specifies a read-only configuration
for the volume. Defaults to false (read/write).
@@ -7596,6 +7809,7 @@ spec:
required:
- fieldPath
type: object
+ x-kubernetes-map-type: atomic
mode:
description: 'Optional: mode bits used to
set permissions on this file, must be an
@@ -7641,6 +7855,7 @@ spec:
required:
- resource
type: object
+ x-kubernetes-map-type: atomic
required:
- path
type: object
@@ -7668,7 +7883,7 @@ spec:
be the minimum value between the SizeLimit specified
here and the sum of memory limits of all containers
in a pod. The default is nil which means that
- the limit is undefined. More info: http://kubernetes.io/docs/user-guide/volumes#emptydir'
+ the limit is undefined. More info: https://kubernetes.io/docs/concepts/storage/volumes#emptydir'
pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$
x-kubernetes-int-or-string: true
type: object
@@ -7792,6 +8007,7 @@ spec:
- kind
- name
type: object
+ x-kubernetes-map-type: atomic
dataSourceRef:
description: 'dataSourceRef specifies the
object from which to populate the volume
@@ -7923,7 +8139,8 @@ spec:
a container, it defaults to Limits
if that is explicitly specified, otherwise
to an implementation-defined value.
- More info: https://kubernetes.io/docs/concepts/configuration/manage-resources-containers/'
+ Requests cannot exceed Limits. More
+ info: https://kubernetes.io/docs/concepts/configuration/manage-resources-containers/'
type: object
type: object
selector:
@@ -7981,6 +8198,7 @@ spec:
ANDed.
type: object
type: object
+ x-kubernetes-map-type: atomic
storageClassName:
description: 'storageClassName is the name
of the StorageClass required by the claim.
@@ -8079,6 +8297,7 @@ spec:
kind, uid?'
type: string
type: object
+ x-kubernetes-map-type: atomic
required:
- driver
type: object
@@ -8269,6 +8488,7 @@ spec:
kind, uid?'
type: string
type: object
+ x-kubernetes-map-type: atomic
targetPortal:
description: targetPortal is iSCSI Target Portal.
The Portal is either an IP or ip_addr:port if
@@ -8451,6 +8671,7 @@ spec:
the ConfigMap or its keys must be defined
type: boolean
type: object
+ x-kubernetes-map-type: atomic
downwardAPI:
description: downwardAPI information about
the downwardAPI data to project
@@ -8482,6 +8703,7 @@ spec:
required:
- fieldPath
type: object
+ x-kubernetes-map-type: atomic
mode:
description: 'Optional: mode bits
used to set permissions on this
@@ -8537,6 +8759,7 @@ spec:
required:
- resource
type: object
+ x-kubernetes-map-type: atomic
required:
- path
type: object
@@ -8608,6 +8831,7 @@ spec:
the Secret or its key must be defined
type: boolean
type: object
+ x-kubernetes-map-type: atomic
serviceAccountToken:
description: serviceAccountToken is information
about the serviceAccountToken data to project
@@ -8734,6 +8958,7 @@ spec:
kind, uid?'
type: string
type: object
+ x-kubernetes-map-type: atomic
user:
description: 'user is the rados user name. Default
is admin. More info: https://examples.k8s.io/volumes/rbd/README.md#how-to-use-it'
@@ -8778,6 +9003,7 @@ spec:
kind, uid?'
type: string
type: object
+ x-kubernetes-map-type: atomic
sslEnabled:
description: sslEnabled Flag enable/disable SSL
communication with Gateway, default false
@@ -8903,6 +9129,7 @@ spec:
kind, uid?'
type: string
type: object
+ x-kubernetes-map-type: atomic
volumeName:
description: volumeName is the human-readable name
of the StorageOS volume. Volume names are only
@@ -9076,6 +9303,7 @@ spec:
- kind
- name
type: object
+ x-kubernetes-map-type: atomic
dataSourceRef:
description: 'dataSourceRef specifies the object from which
to populate the volume with data, if a non-empty volume
@@ -9183,7 +9411,7 @@ spec:
of compute resources required. If Requests is omitted
for a container, it defaults to Limits if that is
explicitly specified, otherwise to an implementation-defined
- value. More info: https://kubernetes.io/docs/concepts/configuration/manage-resources-containers/'
+ value. Requests cannot exceed Limits. More info: https://kubernetes.io/docs/concepts/configuration/manage-resources-containers/'
type: object
type: object
selector:
@@ -9232,6 +9460,7 @@ spec:
contains only "value". The requirements are ANDed.
type: object
type: object
+ x-kubernetes-map-type: atomic
storageClassName:
description: 'storageClassName is the name of the StorageClass
required by the claim. More info: https://kubernetes.io/docs/concepts/storage/persistent-volumes#class-1'
@@ -9256,6 +9485,50 @@ spec:
items:
type: string
type: array
+ allocatedResourceStatuses:
+ additionalProperties:
+ description: When a controller receives persistentvolume
+ claim update with ClaimResourceStatus for a resource
+ that it does not recognizes, then it should ignore that
+ update and let other controllers handle it.
+ type: string
+ description: "allocatedResourceStatuses stores status of
+ resource being resized for the given PVC. Key names follow
+ standard Kubernetes label syntax. Valid values are either:
+ * Un-prefixed keys: - storage - the capacity of the volume.
+ * Custom resources must use implementation-defined prefixed
+ names such as \"example.com/my-custom-resource\" Apart
+ from above values - keys that are unprefixed or have kubernetes.io
+ prefix are considered reserved and hence may not be used.
+ \n ClaimResourceStatus can be in any of following states:
+ - ControllerResizeInProgress: State set when resize controller
+ starts resizing the volume in control-plane. - ControllerResizeFailed:
+ State set when resize has failed in resize controller
+ with a terminal error. - NodeResizePending: State set
+ when resize controller has finished resizing the volume
+ but further resizing of volume is needed on the node.
+ - NodeResizeInProgress: State set when kubelet starts
+ resizing the volume. - NodeResizeFailed: State set when
+ resizing has failed in kubelet with a terminal error.
+ Transient errors don't set NodeResizeFailed. For example:
+ if expanding a PVC for more capacity - this field can
+ be one of the following states: - pvc.status.allocatedResourceStatus['storage']
+ = \"ControllerResizeInProgress\" - pvc.status.allocatedResourceStatus['storage']
+ = \"ControllerResizeFailed\" - pvc.status.allocatedResourceStatus['storage']
+ = \"NodeResizePending\" - pvc.status.allocatedResourceStatus['storage']
+ = \"NodeResizeInProgress\" - pvc.status.allocatedResourceStatus['storage']
+ = \"NodeResizeFailed\" When this field is not set, it
+ means that no resize operation is in progress for the
+ given PVC. \n A controller that receives PVC update with
+ previously unknown resourceName or ClaimResourceStatus
+ should ignore the update for the purpose it was designed.
+ For example - a controller that only is responsible for
+ resizing capacity of the volume, should ignore PVC updates
+ that change other valid resources associated with PVC.
+ \n This is an alpha field and requires enabling RecoverVolumeExpansionFailure
+ feature."
+ type: object
+ x-kubernetes-map-type: granular
allocatedResources:
additionalProperties:
anyOf:
@@ -9263,18 +9536,30 @@ spec:
- type: string
pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$
x-kubernetes-int-or-string: true
- description: allocatedResources is the storage resource
- within AllocatedResources tracks the capacity allocated
- to a PVC. It may be larger than the actual capacity when
- a volume expansion operation is requested. For storage
- quota, the larger value from allocatedResources and PVC.spec.resources
- is used. If allocatedResources is not set, PVC.spec.resources
- alone is used for quota calculation. If a volume expansion
- capacity request is lowered, allocatedResources is only
- lowered if there are no expansion operations in progress
- and if the actual volume capacity is equal or lower than
- the requested capacity. This is an alpha field and requires
- enabling RecoverVolumeExpansionFailure feature.
+ description: "allocatedResources tracks the resources allocated
+ to a PVC including its capacity. Key names follow standard
+ Kubernetes label syntax. Valid values are either: * Un-prefixed
+ keys: - storage - the capacity of the volume. * Custom
+ resources must use implementation-defined prefixed names
+ such as \"example.com/my-custom-resource\" Apart from
+ above values - keys that are unprefixed or have kubernetes.io
+ prefix are considered reserved and hence may not be used.
+ \n Capacity reported here may be larger than the actual
+ capacity when a volume expansion operation is requested.
+ For storage quota, the larger value from allocatedResources
+ and PVC.spec.resources is used. If allocatedResources
+ is not set, PVC.spec.resources alone is used for quota
+ calculation. If a volume expansion capacity request is
+ lowered, allocatedResources is only lowered if there are
+ no expansion operations in progress and if the actual
+ volume capacity is equal or lower than the requested capacity.
+ \n A controller that receives PVC update with previously
+ unknown resourceName should ignore the update for the
+ purpose it was designed. For example - a controller that
+ only is responsible for resizing capacity of the volume,
+ should ignore PVC updates that change other valid resources
+ associated with PVC. \n This is an alpha field and requires
+ enabling RecoverVolumeExpansionFailure feature."
type: object
capacity:
additionalProperties:
@@ -9291,7 +9576,7 @@ spec:
volume claim. If underlying persistent volume is being
resized then the Condition will be set to 'ResizeStarted'.
items:
- description: PersistentVolumeClaimCondition contails details
+ description: PersistentVolumeClaimCondition contains details
about state of pvc
properties:
lastProbeTime:
@@ -9329,13 +9614,6 @@ spec:
phase:
description: phase represents the current phase of PersistentVolumeClaim.
type: string
- resizeStatus:
- description: resizeStatus stores status of resize operation.
- ResizeStatus is not set by default but when expansion
- is complete resizeStatus is set to empty string by resize
- controller or kubelet. This is an alpha field and requires
- enabling RecoverVolumeExpansionFailure feature.
- type: string
type: object
type: object
type: array
diff --git a/config/crd/kustomization.yaml b/config/crd/kustomization.yaml
index c03aa316c33..d6b2f4c3947 100644
--- a/config/crd/kustomization.yaml
+++ b/config/crd/kustomization.yaml
@@ -8,16 +8,15 @@ resources:
- bases/apps.kubeblocks.io_clusterversions.yaml
- bases/apps.kubeblocks.io_configconstraints.yaml
- bases/apps.kubeblocks.io_opsrequests.yaml
-- bases/dataprotection.kubeblocks.io_backuptools.yaml
- bases/dataprotection.kubeblocks.io_backuppolicies.yaml
- bases/dataprotection.kubeblocks.io_backups.yaml
-- bases/dataprotection.kubeblocks.io_restorejobs.yaml
- bases/extensions.kubeblocks.io_addons.yaml
- bases/apps.kubeblocks.io_componentresourceconstraints.yaml
- bases/apps.kubeblocks.io_componentclassdefinitions.yaml
- bases/workloads.kubeblocks.io_replicatedstatemachines.yaml
- bases/storage.kubeblocks.io_storageproviders.yaml
- bases/dataprotection.kubeblocks.io_backuprepos.yaml
+- bases/dataprotection.kubeblocks.io_restores.yaml
- bases/apps.kubeblocks.io_configurations.yaml
- bases/apps.kubeblocks.io_servicedescriptors.yaml
#+kubebuilder:scaffold:crdkustomizeresource
@@ -31,7 +30,6 @@ patchesStrategicMerge:
#- patches/webhook_in_backuptools.yaml
#- patches/webhook_in_backuppolicies.yaml
#- patches/webhook_in_backups.yaml
-#- patches/webhook_in_restorejobs.yaml
#- patches/webhook_in_backuppolicytemplates.yaml
#- patches/webhook_in_opsrequests.yaml
#- patches/webhook_in_reconfigurerequests.yaml
@@ -44,6 +42,7 @@ patchesStrategicMerge:
#- patches/webhook_in_replicatedstatemachines.yaml
#- patches/webhook_in_storageproviders.yaml
#- patches/webhook_in_backuprepos.yaml
+#- patches/webhook_in_restores.yaml
#- patches/webhook_in_configurations.yaml
#- patches/webhook_in_servicedescriptors.yaml
#+kubebuilder:scaffold:crdkustomizewebhookpatch
@@ -56,7 +55,6 @@ patchesStrategicMerge:
#- patches/cainjection_in_backuptools.yaml
#- patches/cainjection_in_backuppolicies.yaml
#- patches/cainjection_in_backups.yaml
-#- patches/cainjection_in_restorejobs.yaml
#- patches/cainjection_in_backuppolicytemplates.yaml
#- patches/cainjection_in_opsrequests.yaml
#- patches/cainjection_in_reconfigurerequests.yaml
@@ -69,6 +67,7 @@ patchesStrategicMerge:
#- patches/cainjection_in_replicatedstatemachines.yaml
#- patches/cainjection_in_storageproviders.yaml
#- patches/cainjection_in_backuprepos.yaml
+#- patches/cainjection_in_restores.yaml
#- patches/cainjection_in_configurations.yaml
#- patches/cainjection_in_servicedescriptors.yaml
#+kubebuilder:scaffold:crdkustomizecainjectionpatch
diff --git a/config/crd/patches/cainjection_in_dataprotection_restorejobs.yaml b/config/crd/patches/cainjection_in_restores.yaml
similarity index 83%
rename from config/crd/patches/cainjection_in_dataprotection_restorejobs.yaml
rename to config/crd/patches/cainjection_in_restores.yaml
index f8863965617..dc4c069a797 100644
--- a/config/crd/patches/cainjection_in_dataprotection_restorejobs.yaml
+++ b/config/crd/patches/cainjection_in_restores.yaml
@@ -4,4 +4,4 @@ kind: CustomResourceDefinition
metadata:
annotations:
cert-manager.io/inject-ca-from: $(CERTIFICATE_NAMESPACE)/$(CERTIFICATE_NAME)
- name: restorejobs.dataprotection.kubeblocks.io
+ name: restores.dataprotection.kubeblocks.io
diff --git a/config/crd/patches/webhook_in_dataprotection_restorejobs.yaml b/config/crd/patches/webhook_in_restores.yaml
similarity index 88%
rename from config/crd/patches/webhook_in_dataprotection_restorejobs.yaml
rename to config/crd/patches/webhook_in_restores.yaml
index c138579306e..6816d399a6f 100644
--- a/config/crd/patches/webhook_in_dataprotection_restorejobs.yaml
+++ b/config/crd/patches/webhook_in_restores.yaml
@@ -2,7 +2,7 @@
apiVersion: apiextensions.k8s.io/v1
kind: CustomResourceDefinition
metadata:
- name: restorejobs.dataprotection.kubeblocks.io
+ name: restores.dataprotection.kubeblocks.io
spec:
conversion:
strategy: Webhook
diff --git a/config/rbac/dataprotection_restorejob_editor_role.yaml b/config/rbac/dataprotection_estore_editor_role.yaml
similarity index 70%
rename from config/rbac/dataprotection_restorejob_editor_role.yaml
rename to config/rbac/dataprotection_estore_editor_role.yaml
index 8b7fad4d512..e232ebedc31 100644
--- a/config/rbac/dataprotection_restorejob_editor_role.yaml
+++ b/config/rbac/dataprotection_estore_editor_role.yaml
@@ -1,13 +1,13 @@
-# permissions for end users to edit restorejobs.
+# permissions for end users to edit restores.
apiVersion: rbac.authorization.k8s.io/v1
kind: ClusterRole
metadata:
- name: restorejob-editor-role
+ name: restore-editor-role
rules:
- apiGroups:
- dataprotection.kubeblocks.io
resources:
- - restorejobs
+ - restores
verbs:
- create
- delete
@@ -19,6 +19,6 @@ rules:
- apiGroups:
- dataprotection.kubeblocks.io
resources:
- - restorejobs/status
+ - restores/status
verbs:
- get
diff --git a/config/rbac/dataprotection_restorejob_viewer_role.yaml b/config/rbac/dataprotection_restore_viewer_role.yaml
similarity index 67%
rename from config/rbac/dataprotection_restorejob_viewer_role.yaml
rename to config/rbac/dataprotection_restore_viewer_role.yaml
index e11a21eda8c..ce5953eb959 100644
--- a/config/rbac/dataprotection_restorejob_viewer_role.yaml
+++ b/config/rbac/dataprotection_restore_viewer_role.yaml
@@ -1,13 +1,13 @@
-# permissions for end users to view restorejobs.
+# permissions for end users to view restores.
apiVersion: rbac.authorization.k8s.io/v1
kind: ClusterRole
metadata:
- name: restorejob-viewer-role
+ name: restore-viewer-role
rules:
- apiGroups:
- dataprotection.kubeblocks.io
resources:
- - restorejobs
+ - restores
verbs:
- get
- list
@@ -15,6 +15,6 @@ rules:
- apiGroups:
- dataprotection.kubeblocks.io
resources:
- - restorejobs/status
+ - restores/status
verbs:
- get
diff --git a/config/rbac/role.yaml b/config/rbac/role.yaml
index e43058fbb5a..57454b4073c 100644
--- a/config/rbac/role.yaml
+++ b/config/rbac/role.yaml
@@ -2,7 +2,6 @@
apiVersion: rbac.authorization.k8s.io/v1
kind: ClusterRole
metadata:
- creationTimestamp: null
name: manager-role
rules:
- apiGroups:
@@ -567,6 +566,32 @@ rules:
- services/status
verbs:
- get
+- apiGroups:
+ - dataprotection.kubeblocks.io
+ resources:
+ - actionsets
+ verbs:
+ - create
+ - delete
+ - get
+ - list
+ - patch
+ - update
+ - watch
+- apiGroups:
+ - dataprotection.kubeblocks.io
+ resources:
+ - actionsets/finalizers
+ verbs:
+ - update
+- apiGroups:
+ - dataprotection.kubeblocks.io
+ resources:
+ - actionsets/status
+ verbs:
+ - get
+ - patch
+ - update
- apiGroups:
- dataprotection.kubeblocks.io
resources:
@@ -650,7 +675,7 @@ rules:
- apiGroups:
- dataprotection.kubeblocks.io
resources:
- - backuptools
+ - backupschedules
verbs:
- create
- delete
@@ -662,13 +687,13 @@ rules:
- apiGroups:
- dataprotection.kubeblocks.io
resources:
- - backuptools/finalizers
+ - backupschedules/finalizers
verbs:
- update
- apiGroups:
- dataprotection.kubeblocks.io
resources:
- - backuptools/status
+ - backupschedules/status
verbs:
- get
- patch
@@ -676,7 +701,7 @@ rules:
- apiGroups:
- dataprotection.kubeblocks.io
resources:
- - restorejobs
+ - restores
verbs:
- create
- delete
@@ -688,13 +713,13 @@ rules:
- apiGroups:
- dataprotection.kubeblocks.io
resources:
- - restorejobs/finalizers
+ - restores/finalizers
verbs:
- update
- apiGroups:
- dataprotection.kubeblocks.io
resources:
- - restorejobs/status
+ - restores/status
verbs:
- get
- patch
diff --git a/config/samples/dataprotection_v1alpha1_restore.yaml b/config/samples/dataprotection_v1alpha1_restore.yaml
new file mode 100644
index 00000000000..db589a76597
--- /dev/null
+++ b/config/samples/dataprotection_v1alpha1_restore.yaml
@@ -0,0 +1,6 @@
+apiVersion: dataprotection.kubeblocks.io/v1alpha1
+kind: Restore
+metadata:
+ name: restore-sample
+spec:
+ # TODO(user): Add fields here
diff --git a/config/webhook/manifests.yaml b/config/webhook/manifests.yaml
index 192f52d5806..562dc0d1d77 100644
--- a/config/webhook/manifests.yaml
+++ b/config/webhook/manifests.yaml
@@ -2,9 +2,28 @@
apiVersion: admissionregistration.k8s.io/v1
kind: MutatingWebhookConfiguration
metadata:
- creationTimestamp: null
name: mutating-webhook-configuration
webhooks:
+- admissionReviewVersions:
+ - v1
+ clientConfig:
+ service:
+ name: webhook-service
+ namespace: system
+ path: /mutate-workloads-kubeblocks-io-v1alpha1-replicatedstatemachine
+ failurePolicy: Fail
+ name: mreplicatedstatemachine.kb.io
+ rules:
+ - apiGroups:
+ - workloads.kubeblocks.io
+ apiVersions:
+ - v1alpha1
+ operations:
+ - CREATE
+ - UPDATE
+ resources:
+ - replicatedstatemachines
+ sideEffects: None
- admissionReviewVersions:
- v1
clientConfig:
@@ -45,15 +64,21 @@ webhooks:
resources:
- servicedescriptors
sideEffects: None
+---
+apiVersion: admissionregistration.k8s.io/v1
+kind: ValidatingWebhookConfiguration
+metadata:
+ name: validating-webhook-configuration
+webhooks:
- admissionReviewVersions:
- v1
clientConfig:
service:
name: webhook-service
namespace: system
- path: /mutate-workloads-kubeblocks-io-v1alpha1-replicatedstatemachine
+ path: /validate-workloads-kubeblocks-io-v1alpha1-replicatedstatemachine
failurePolicy: Fail
- name: mreplicatedstatemachine.kb.io
+ name: vreplicatedstatemachine.kb.io
rules:
- apiGroups:
- workloads.kubeblocks.io
@@ -65,13 +90,6 @@ webhooks:
resources:
- replicatedstatemachines
sideEffects: None
----
-apiVersion: admissionregistration.k8s.io/v1
-kind: ValidatingWebhookConfiguration
-metadata:
- creationTimestamp: null
- name: validating-webhook-configuration
-webhooks:
- admissionReviewVersions:
- v1
clientConfig:
@@ -172,23 +190,3 @@ webhooks:
resources:
- servicedescriptors
sideEffects: None
-- admissionReviewVersions:
- - v1
- clientConfig:
- service:
- name: webhook-service
- namespace: system
- path: /validate-workloads-kubeblocks-io-v1alpha1-replicatedstatemachine
- failurePolicy: Fail
- name: vreplicatedstatemachine.kb.io
- rules:
- - apiGroups:
- - workloads.kubeblocks.io
- apiVersions:
- - v1alpha1
- operations:
- - CREATE
- - UPDATE
- resources:
- - replicatedstatemachines
- sideEffects: None
diff --git a/controllers/apps/class_controller_test.go b/controllers/apps/class_controller_test.go
index 25a427d2f05..194480afbfd 100644
--- a/controllers/apps/class_controller_test.go
+++ b/controllers/apps/class_controller_test.go
@@ -22,6 +22,7 @@ package apps
import (
. "github.com/onsi/ginkgo/v2"
. "github.com/onsi/gomega"
+
"sigs.k8s.io/controller-runtime/pkg/client"
"github.com/apecloud/kubeblocks/apis/apps/v1alpha1"
diff --git a/controllers/apps/cluster_controller.go b/controllers/apps/cluster_controller.go
index 522f8b66c35..9e0cec3815e 100644
--- a/controllers/apps/cluster_controller.go
+++ b/controllers/apps/cluster_controller.go
@@ -36,10 +36,9 @@ import (
"sigs.k8s.io/controller-runtime/pkg/handler"
"sigs.k8s.io/controller-runtime/pkg/log"
"sigs.k8s.io/controller-runtime/pkg/reconcile"
- "sigs.k8s.io/controller-runtime/pkg/source"
appsv1alpha1 "github.com/apecloud/kubeblocks/apis/apps/v1alpha1"
- dataprotectionv1alpha1 "github.com/apecloud/kubeblocks/apis/dataprotection/v1alpha1"
+ dpv1alpha1 "github.com/apecloud/kubeblocks/apis/dataprotection/v1alpha1"
workloads "github.com/apecloud/kubeblocks/apis/workloads/v1alpha1"
"github.com/apecloud/kubeblocks/internal/constant"
intctrlutil "github.com/apecloud/kubeblocks/internal/controllerutil"
@@ -180,12 +179,12 @@ func (r *ClusterReconciler) Reconcile(ctx context.Context, req ctrl.Request) (ct
&ValidateEnableLogsTransformer{},
// create cluster connection credential secret object
&ClusterCredentialTransformer{},
- // handle restore
+ // handle restore before ComponentTransformer
&RestoreTransformer{Client: r.Client},
// create all components objects
&ComponentTransformer{Client: r.Client},
// transform backupPolicy tpl to backuppolicy.dataprotection.kubeblocks.io
- &BackupPolicyTPLTransformer{},
+ &BackupPolicyTplTransformer{},
// handle rbac for pod
&RBACTransformer{},
// add our finalizer to all objects
@@ -227,25 +226,27 @@ func (r *ClusterReconciler) SetupWithManager(mgr ctrl.Manager) error {
Owns(&corev1.ConfigMap{}).
Owns(&corev1.PersistentVolumeClaim{}).
Owns(&policyv1.PodDisruptionBudget{}).
- Owns(&dataprotectionv1alpha1.BackupPolicy{}).
- Owns(&dataprotectionv1alpha1.Backup{}).
+ Owns(&dpv1alpha1.BackupPolicy{}).
+ Owns(&dpv1alpha1.BackupSchedule{}).
+ Owns(&dpv1alpha1.Backup{}).
+ Owns(&dpv1alpha1.Restore{}).
Owns(&batchv1.Job{}).
- Watches(&source.Kind{Type: &corev1.Pod{}}, handler.EnqueueRequestsFromMapFunc(r.filterClusterResources))
+ Watches(&corev1.Pod{}, handler.EnqueueRequestsFromMapFunc(r.filterClusterResources))
if viper.GetBool(constant.EnableRBACManager) {
b.Owns(&rbacv1.ClusterRoleBinding{}).
Owns(&rbacv1.RoleBinding{}).
Owns(&corev1.ServiceAccount{})
} else {
- b.Watches(&source.Kind{Type: &rbacv1.ClusterRoleBinding{}}, handler.EnqueueRequestsFromMapFunc(r.filterClusterResources)).
- Watches(&source.Kind{Type: &rbacv1.RoleBinding{}}, handler.EnqueueRequestsFromMapFunc(r.filterClusterResources)).
- Watches(&source.Kind{Type: &corev1.ServiceAccount{}}, handler.EnqueueRequestsFromMapFunc(r.filterClusterResources))
+ b.Watches(&rbacv1.ClusterRoleBinding{}, handler.EnqueueRequestsFromMapFunc(r.filterClusterResources)).
+ Watches(&rbacv1.RoleBinding{}, handler.EnqueueRequestsFromMapFunc(r.filterClusterResources)).
+ Watches(&corev1.ServiceAccount{}, handler.EnqueueRequestsFromMapFunc(r.filterClusterResources))
}
return b.Complete(r)
}
-func (r *ClusterReconciler) filterClusterResources(obj client.Object) []reconcile.Request {
+func (r *ClusterReconciler) filterClusterResources(ctx context.Context, obj client.Object) []reconcile.Request {
labels := obj.GetLabels()
if v, ok := labels[constant.AppManagedByLabelKey]; !ok || v != constant.AppName {
return []reconcile.Request{}
diff --git a/controllers/apps/cluster_controller_test.go b/controllers/apps/cluster_controller_test.go
index 3c8fc176107..d694b534c78 100644
--- a/controllers/apps/cluster_controller_test.go
+++ b/controllers/apps/cluster_controller_test.go
@@ -29,13 +29,13 @@ import (
"strings"
"time"
- snapshotv1 "github.com/kubernetes-csi/external-snapshotter/client/v6/apis/volumesnapshot/v1"
. "github.com/onsi/ginkgo/v2"
. "github.com/onsi/gomega"
+
+ snapshotv1 "github.com/kubernetes-csi/external-snapshotter/client/v6/apis/volumesnapshot/v1"
"github.com/sethvargo/go-password/password"
"golang.org/x/exp/slices"
appsv1 "k8s.io/api/apps/v1"
- batchv1 "k8s.io/api/batch/v1"
corev1 "k8s.io/api/core/v1"
rbacv1 "k8s.io/api/rbac/v1"
storagev1 "k8s.io/api/storage/v1"
@@ -51,21 +51,27 @@ import (
"sigs.k8s.io/controller-runtime/pkg/controller/controllerutil"
appsv1alpha1 "github.com/apecloud/kubeblocks/apis/apps/v1alpha1"
- dataprotectionv1alpha1 "github.com/apecloud/kubeblocks/apis/dataprotection/v1alpha1"
+ dpv1alpha1 "github.com/apecloud/kubeblocks/apis/dataprotection/v1alpha1"
workloads "github.com/apecloud/kubeblocks/apis/workloads/v1alpha1"
"github.com/apecloud/kubeblocks/controllers/apps/components"
+ "github.com/apecloud/kubeblocks/internal/common"
"github.com/apecloud/kubeblocks/internal/constant"
intctrlutil "github.com/apecloud/kubeblocks/internal/controllerutil"
+ dptypes "github.com/apecloud/kubeblocks/internal/dataprotection/types"
"github.com/apecloud/kubeblocks/internal/generics"
testapps "github.com/apecloud/kubeblocks/internal/testutil/apps"
+ testdp "github.com/apecloud/kubeblocks/internal/testutil/dataprotection"
testk8s "github.com/apecloud/kubeblocks/internal/testutil/k8s"
viper "github.com/apecloud/kubeblocks/internal/viperx"
lorry "github.com/apecloud/kubeblocks/lorry/client"
- lorryutil "github.com/apecloud/kubeblocks/lorry/util"
)
const (
backupPolicyTPLName = "test-backup-policy-template-mysql"
+ backupMethodName = "test-backup-method"
+ vsBackupMethodName = "test-vs-backup-method"
+ actionSetName = "test-action-set"
+ vsActionSetName = "test-vs-action-set"
)
var (
@@ -125,7 +131,7 @@ var _ = Describe("Cluster Controller", func() {
consensusCompDefName = "consensus"
replicationCompName = "replication"
replicationCompDefName = "replication"
- backupToolName = "test-backup-tool"
+ actionSetName = "test-actionset"
)
var (
@@ -174,12 +180,12 @@ var _ = Describe("Cluster Controller", func() {
// namespaced
testapps.ClearResourcesWithRemoveFinalizerOption(&testCtx, generics.PersistentVolumeClaimSignature, true, inNS, ml)
testapps.ClearResourcesWithRemoveFinalizerOption(&testCtx, generics.PodSignature, true, inNS, ml)
- testapps.ClearResources(&testCtx, generics.BackupSignature, inNS, ml)
- testapps.ClearResources(&testCtx, generics.BackupPolicySignature, inNS, ml)
+ testapps.ClearResourcesWithRemoveFinalizerOption(&testCtx, generics.BackupSignature, true, inNS, ml)
+ testapps.ClearResourcesWithRemoveFinalizerOption(&testCtx, generics.BackupPolicySignature, true, inNS, ml)
testapps.ClearResourcesWithRemoveFinalizerOption(&testCtx, generics.VolumeSnapshotSignature, true, inNS)
// non-namespaced
testapps.ClearResources(&testCtx, generics.BackupPolicyTemplateSignature, ml)
- testapps.ClearResources(&testCtx, generics.BackupToolSignature, ml)
+ testapps.ClearResources(&testCtx, generics.ActionSetSignature, ml)
testapps.ClearResources(&testCtx, generics.StorageClassSignature, ml)
resetTestContext()
}
@@ -378,9 +384,10 @@ var _ = Describe("Cluster Controller", func() {
By("Mocking a retained backup")
backupPolicyName := "test-backup-policy"
backupName := "test-backup"
- backup := testapps.NewBackupFactory(testCtx.DefaultNamespace, backupName).
+ backupMethod := "test-backup-method"
+ backup := testdp.NewBackupFactory(testCtx.DefaultNamespace, backupName).
SetBackupPolicyName(backupPolicyName).
- SetBackupType(dataprotectionv1alpha1.BackupTypeDataFile).
+ SetBackupMethod(backupMethod).
SetLabels(map[string]string{constant.AppInstanceLabelKey: clusterKey.Name, constant.BackupProtectionLabelKey: constant.BackupRetain}).
WithRandomName().
Create(&testCtx).GetObject()
@@ -395,7 +402,7 @@ var _ = Describe("Cluster Controller", func() {
Eventually(testapps.CheckObjExists(&testCtx, clusterKey, &appsv1alpha1.Cluster{}, false)).Should(Succeed())
By("Checking backup should exist")
- Eventually(testapps.CheckObjExists(&testCtx, backupKey, &dataprotectionv1alpha1.Backup{}, true)).Should(Succeed())
+ Eventually(testapps.CheckObjExists(&testCtx, backupKey, &dpv1alpha1.Backup{}, true)).Should(Succeed())
}
testDoNotTerminate := func(compName, compDefName string) {
@@ -449,36 +456,11 @@ var _ = Describe("Cluster Controller", func() {
}
checkSingleWorkload := func(compDefName string, expects func(g Gomega, sts *appsv1.StatefulSet, deploy *appsv1.Deployment)) {
- if intctrlutil.IsRSMEnabled() {
- Eventually(func(g Gomega) {
- l := testk8s.ListAndCheckRSM(&testCtx, clusterKey)
- sts := components.ConvertRSMToSTS(&l.Items[0])
- expects(g, sts, nil)
- }).Should(Succeed())
- return
- }
-
- isStsWorkload := true
- switch compDefName {
- case statelessCompDefName:
- isStsWorkload = false
- case statefulCompDefName, replicationCompDefName, consensusCompDefName:
- break
- default:
- panic("unreachable")
- }
-
- if isStsWorkload {
- Eventually(func(g Gomega) {
- l := testk8s.ListAndCheckStatefulSet(&testCtx, clusterKey)
- expects(g, &l.Items[0], nil)
- }).Should(Succeed())
- } else {
- Eventually(func(g Gomega) {
- l := testk8s.ListAndCheckDeployment(&testCtx, clusterKey)
- expects(g, nil, &l.Items[0])
- }).Should(Succeed())
- }
+ Eventually(func(g Gomega) {
+ l := testk8s.ListAndCheckRSM(&testCtx, clusterKey)
+ sts := components.ConvertRSMToSTS(&l.Items[0])
+ expects(g, sts, nil)
+ }).Should(Succeed())
}
testChangeReplicas := func(compName, compDefName string) {
@@ -510,16 +492,24 @@ var _ = Describe("Cluster Controller", func() {
return fmt.Sprintf("%s-%s-%s-%d", vctName, clusterKey.Name, compName, i)
}
- createPVC := func(clusterName, pvcName, compName string) {
+ createPVC := func(clusterName, pvcName, compName, storageSize, storageClassName string) {
+ if storageSize == "" {
+ storageSize = "1Gi"
+ }
+ clusterBytes, _ := json.Marshal(clusterObj)
testapps.NewPersistentVolumeClaimFactory(testCtx.DefaultNamespace, pvcName, clusterName,
- compName, "data").SetStorage("1Gi").AddLabelsInMap(map[string]string{
- constant.AppInstanceLabelKey: clusterName,
- constant.KBAppComponentLabelKey: compName,
- constant.AppManagedByLabelKey: constant.AppName,
- }).CheckedCreate(&testCtx)
+ compName, testapps.DataVolumeName).
+ AddLabelsInMap(map[string]string{
+ constant.AppInstanceLabelKey: clusterName,
+ constant.KBAppComponentLabelKey: compName,
+ constant.AppManagedByLabelKey: constant.AppName,
+ }).AddAnnotations(constant.LastAppliedClusterAnnotationKey, string(clusterBytes)).
+ SetStorage(storageSize).
+ SetStorageClass(storageClassName).
+ CheckedCreate(&testCtx)
}
- mockComponentPVCsBound := func(comp *appsv1alpha1.ClusterComponentSpec, replicas int, create bool) {
+ mockComponentPVCsAndBound := func(comp *appsv1alpha1.ClusterComponentSpec, replicas int, create bool, storageClassName string) {
for i := 0; i < replicas; i++ {
for _, vct := range comp.VolumeClaimTemplates {
pvcKey := types.NamespacedName{
@@ -527,7 +517,7 @@ var _ = Describe("Cluster Controller", func() {
Name: getPVCName(vct.Name, comp.Name, i),
}
if create {
- createPVC(clusterKey.Name, pvcKey.Name, comp.Name)
+ createPVC(clusterKey.Name, pvcKey.Name, comp.Name, vct.Spec.Resources.Requests.Storage().String(), storageClassName)
}
Eventually(testapps.CheckObjExists(&testCtx, pvcKey,
&corev1.PersistentVolumeClaim{}, true)).Should(Succeed())
@@ -576,19 +566,14 @@ var _ = Describe("Cluster Controller", func() {
return pods
}
- horizontalScaleComp := func(updatedReplicas int, comp *appsv1alpha1.ClusterComponentSpec, policy *appsv1alpha1.HorizontalScalePolicy) {
+ horizontalScaleComp := func(updatedReplicas int, comp *appsv1alpha1.ClusterComponentSpec,
+ storageClassName string, policy *appsv1alpha1.HorizontalScalePolicy) {
By("Mocking component PVCs to bound")
- mockComponentPVCsBound(comp, int(comp.Replicas), true)
+ mockComponentPVCsAndBound(comp, int(comp.Replicas), true, storageClassName)
- if intctrlutil.IsRSMEnabled() {
- By("Checking rsm replicas right")
- rsmList := testk8s.ListAndCheckRSMWithComponent(&testCtx, clusterKey, comp.Name)
- Expect(int(*rsmList.Items[0].Spec.Replicas)).To(BeEquivalentTo(comp.Replicas))
- } else {
- By("Checking sts replicas right")
- stsList := testk8s.ListAndCheckStatefulSetWithComponent(&testCtx, clusterKey, comp.Name)
- Expect(int(*stsList.Items[0].Spec.Replicas)).To(BeEquivalentTo(comp.Replicas))
- }
+ By("Checking rsm replicas right")
+ rsmList := testk8s.ListAndCheckRSMWithComponent(&testCtx, clusterKey, comp.Name)
+ Expect(int(*rsmList.Items[0].Spec.Replicas)).To(BeEquivalentTo(comp.Replicas))
By("Creating mock pods in StatefulSet")
pods := mockPodsForTest(clusterObj, int(comp.Replicas))
@@ -612,12 +597,8 @@ var _ = Describe("Cluster Controller", func() {
checkUpdatedStsReplicas := func() {
By("Checking updated sts replicas")
Eventually(func() int32 {
- if intctrlutil.IsRSMEnabled() {
- rsmList := testk8s.ListAndCheckRSMWithComponent(&testCtx, clusterKey, comp.Name)
- return *rsmList.Items[0].Spec.Replicas
- }
- stsList := testk8s.ListAndCheckStatefulSetWithComponent(&testCtx, clusterKey, comp.Name)
- return *stsList.Items[0].Spec.Replicas
+ rsmList := testk8s.ListAndCheckRSMWithComponent(&testCtx, clusterKey, comp.Name)
+ return *rsmList.Items[0].Spec.Replicas
}).Should(BeEquivalentTo(updatedReplicas))
}
@@ -626,25 +607,27 @@ var _ = Describe("Cluster Controller", func() {
return
}
+ ml := client.MatchingLabels{
+ constant.AppInstanceLabelKey: clusterKey.Name,
+ constant.KBAppComponentLabelKey: comp.Name,
+ constant.KBManagedByKey: "cluster",
+ }
if policy != nil {
By(fmt.Sprintf("Checking backup of component %s created", comp.Name))
Eventually(testapps.List(&testCtx, generics.BackupSignature,
- client.MatchingLabels{
- constant.AppInstanceLabelKey: clusterKey.Name,
- constant.KBAppComponentLabelKey: comp.Name,
- }, client.InNamespace(clusterKey.Namespace))).Should(HaveLen(1))
+ ml, client.InNamespace(clusterKey.Namespace))).Should(HaveLen(1))
backupKey := types.NamespacedName{Name: fmt.Sprintf("%s-%s-scaling",
clusterKey.Name, comp.Name),
Namespace: testCtx.DefaultNamespace}
By("Mocking backup status to completed")
- Expect(testapps.GetAndChangeObjStatus(&testCtx, backupKey, func(backup *dataprotectionv1alpha1.Backup) {
- backup.Status.Phase = dataprotectionv1alpha1.BackupCompleted
+ Expect(testapps.GetAndChangeObjStatus(&testCtx, backupKey, func(backup *dpv1alpha1.Backup) {
+ backup.Status.Phase = dpv1alpha1.BackupPhaseCompleted
backup.Status.PersistentVolumeClaimName = "backup-data"
- backup.Status.BackupToolName = backupToolName
+ testdp.MockBackupStatusMethod(backup, testapps.DataVolumeName)
})()).Should(Succeed())
- if viper.GetBool("VOLUMESNAPSHOT") {
+ if testk8s.IsMockVolumeSnapshotEnabled(&testCtx, storageClassName) {
By("Mocking VolumeSnapshot and set it as ReadyToUse")
pvcName := getPVCName(testapps.DataVolumeName, comp.Name, 0)
volumeSnapshot := &snapshotv1.VolumeSnapshot{
@@ -652,7 +635,7 @@ var _ = Describe("Cluster Controller", func() {
Name: backupKey.Name,
Namespace: backupKey.Namespace,
Labels: map[string]string{
- constant.DataProtectionLabelBackupNameKey: backupKey.Name,
+ dptypes.DataProtectionLabelBackupNameKey: backupKey.Name,
}},
Spec: snapshotv1.VolumeSnapshotSpec{
Source: snapshotv1.VolumeSnapshotSource{
@@ -670,59 +653,17 @@ var _ = Describe("Cluster Controller", func() {
}
}
- By("Checking pvc created")
- Eventually(testapps.List(&testCtx, generics.PersistentVolumeClaimSignature,
- client.MatchingLabels{
- constant.AppInstanceLabelKey: clusterKey.Name,
- constant.KBAppComponentLabelKey: comp.Name,
- }, client.InNamespace(clusterKey.Namespace))).Should(HaveLen(updatedReplicas * len(comp.VolumeClaimTemplates)))
+ By("Mock PVCs and set status to bound")
+ mockComponentPVCsAndBound(comp, updatedReplicas, true, storageClassName)
if policy != nil {
- if !viper.GetBool("VOLUMESNAPSHOT") && len(viper.GetString(constant.CfgKeyBackupPVCName)) > 0 {
- By("Checking restore job created")
- Eventually(testapps.List(&testCtx, generics.JobSignature,
- client.MatchingLabels{
- constant.AppInstanceLabelKey: clusterKey.Name,
- constant.KBAppComponentLabelKey: comp.Name,
- constant.KBManagedByKey: "cluster",
- }, client.InNamespace(clusterKey.Namespace))).Should(HaveLen(updatedReplicas - int(comp.Replicas)))
-
- By("Mocking job status to succeeded")
- ml := client.MatchingLabels{
- constant.AppInstanceLabelKey: clusterKey.Name,
- constant.KBAppComponentLabelKey: comp.Name,
- constant.KBManagedByKey: "cluster",
- }
- jobList := batchv1.JobList{}
- Expect(testCtx.Cli.List(testCtx.Ctx, &jobList, ml)).Should(Succeed())
- for _, job := range jobList.Items {
- key := client.ObjectKeyFromObject(&job)
- Expect(testapps.GetAndChangeObjStatus(&testCtx, key, func(job *batchv1.Job) {
- job.Status.Succeeded = 1
- })()).Should(Succeed())
- }
- }
+ checkRestoreAndSetCompleted(clusterKey, comp.Name, updatedReplicas-int(comp.Replicas))
}
- By("Mock PVCs status to bound")
- mockComponentPVCsBound(comp, updatedReplicas, false)
-
if policy != nil {
- By("Checking backup job cleanup")
- Eventually(testapps.List(&testCtx, generics.BackupSignature,
- client.MatchingLabels{
- constant.AppInstanceLabelKey: clusterKey.Name,
- constant.KBAppComponentLabelKey: comp.Name,
- }, client.InNamespace(clusterKey.Namespace))).Should(HaveLen(0))
-
- if !viper.GetBool("VOLUMESNAPSHOT") && len(viper.GetString(constant.CfgKeyBackupPVCName)) > 0 {
- By("Checking restore job cleanup")
- Eventually(testapps.List(&testCtx, generics.JobSignature,
- client.MatchingLabels{
- constant.AppInstanceLabelKey: clusterKey.Name,
- constant.KBAppComponentLabelKey: comp.Name,
- }, client.InNamespace(clusterKey.Namespace))).Should(HaveLen(0))
- }
+ By("Checking Backup and Restore cleanup")
+ Eventually(testapps.List(&testCtx, generics.BackupSignature, ml, client.InNamespace(clusterKey.Namespace))).Should(HaveLen(0))
+ Eventually(testapps.List(&testCtx, generics.RestoreSignature, ml, client.InNamespace(clusterKey.Namespace))).Should(HaveLen(0))
}
checkUpdatedStsReplicas()
@@ -744,38 +685,6 @@ var _ = Describe("Cluster Controller", func() {
})).Should(Succeed())
}
}
-
- if !intctrlutil.IsRSMEnabled() {
- By("Checking pod env config updated")
- cmKey := types.NamespacedName{
- Namespace: clusterKey.Namespace,
- Name: fmt.Sprintf("%s-%s-env", clusterKey.Name, comp.Name),
- }
- Eventually(testapps.CheckObj(&testCtx, cmKey, func(g Gomega, cm *corev1.ConfigMap) {
- match := func(key, prefix, suffix string) bool {
- return strings.HasPrefix(key, prefix) && strings.HasSuffix(key, suffix)
- }
- foundN := ""
- for k, v := range cm.Data {
- if match(k, constant.KBPrefix, "_N") {
- foundN = v
- break
- }
- }
- g.Expect(foundN).Should(Equal(strconv.Itoa(updatedReplicas)))
- for i := 0; i < updatedReplicas; i++ {
- foundPodHostname := ""
- suffix := fmt.Sprintf("_%d_HOSTNAME", i)
- for k, v := range cm.Data {
- if match(k, constant.KBPrefix, suffix) {
- foundPodHostname = v
- break
- }
- }
- g.Expect(foundPodHostname != "").Should(BeTrue())
- }
- })).Should(Succeed())
- }
}
scaleInCheck := func() {
@@ -862,37 +771,45 @@ var _ = Describe("Cluster Controller", func() {
}
By("Checking backup policy created from backup policy template")
- policyName := DeriveBackupPolicyName(clusterKey.Name, compDef.Name, "")
+ policyName := generateBackupPolicyName(clusterKey.Name, compDef.Name, "")
clusterDef.Spec.ComponentDefs[i].HorizontalScalePolicy = &appsv1alpha1.HorizontalScalePolicy{
Type: policyType,
BackupPolicyTemplateName: backupPolicyTPLName,
}
Eventually(testapps.CheckObjExists(&testCtx, client.ObjectKey{Name: policyName, Namespace: clusterKey.Namespace},
- &dataprotectionv1alpha1.BackupPolicy{}, true)).Should(Succeed())
+ &dpv1alpha1.BackupPolicy{}, true)).Should(Succeed())
if policyType == appsv1alpha1.HScaleDataClonePolicyCloneVolume {
- By("creating backup tool if backup policy is backup")
- backupTool := &dataprotectionv1alpha1.BackupTool{
+ By("creating actionSet if backup policy is backup")
+ actionSet := &dpv1alpha1.ActionSet{
ObjectMeta: metav1.ObjectMeta{
- Name: backupToolName,
+ Name: actionSetName,
Namespace: clusterKey.Namespace,
Labels: map[string]string{
constant.ClusterDefLabelKey: clusterDef.Name,
},
},
- Spec: dataprotectionv1alpha1.BackupToolSpec{
- BackupCommands: []string{""},
- Image: "xtrabackup",
+ Spec: dpv1alpha1.ActionSetSpec{
Env: []corev1.EnvVar{
{
Name: "test-name",
Value: "test-value",
},
},
- Physical: &dataprotectionv1alpha1.PhysicalConfig{
- BackupToolRestoreCommand: dataprotectionv1alpha1.BackupToolRestoreCommand{
- RestoreCommands: []string{
+ BackupType: dpv1alpha1.BackupTypeFull,
+ Backup: &dpv1alpha1.BackupActionSpec{
+ BackupData: &dpv1alpha1.BackupDataActionSpec{
+ JobActionSpec: dpv1alpha1.JobActionSpec{
+ Image: "xtrabackup",
+ Command: []string{""},
+ },
+ },
+ },
+ Restore: &dpv1alpha1.RestoreActionSpec{
+ PrepareData: &dpv1alpha1.JobActionSpec{
+ Image: "xtrabackup",
+ Command: []string{
"sh",
"-c",
"/backup_scripts.sh",
@@ -901,7 +818,7 @@ var _ = Describe("Cluster Controller", func() {
},
},
}
- testapps.CheckedCreateK8sResource(&testCtx, backupTool)
+ testapps.CheckedCreateK8sResource(&testCtx, actionSet)
}
}
})()).ShouldNot(HaveOccurred())
@@ -909,7 +826,8 @@ var _ = Describe("Cluster Controller", func() {
// @argument componentDefsWithHScalePolicy assign ClusterDefinition.spec.componentDefs[].horizontalScalePolicy for
// the matching names. If not provided, will set 1st ClusterDefinition.spec.componentDefs[0].horizontalScalePolicy.
- horizontalScale := func(updatedReplicas int, policyType appsv1alpha1.HScaleDataClonePolicyType, componentDefsWithHScalePolicy ...string) {
+ horizontalScale := func(updatedReplicas int, storageClassName string,
+ policyType appsv1alpha1.HScaleDataClonePolicyType, componentDefsWithHScalePolicy ...string) {
defer lorry.UnsetMockClient()
cluster := &appsv1alpha1.Cluster{}
@@ -919,8 +837,8 @@ var _ = Describe("Cluster Controller", func() {
setHorizontalScalePolicy(policyType, componentDefsWithHScalePolicy...)
By("Mocking all components' PVCs to bound")
- for _, comp := range clusterObj.Spec.ComponentSpecs {
- mockComponentPVCsBound(&comp, int(comp.Replicas), true)
+ for _, comp := range cluster.Spec.ComponentSpecs {
+ mockComponentPVCsAndBound(&comp, int(comp.Replicas), true, storageClassName)
}
hscalePolicy := func(comp appsv1alpha1.ClusterComponentSpec) *appsv1alpha1.HorizontalScalePolicy {
@@ -934,16 +852,16 @@ var _ = Describe("Cluster Controller", func() {
By("Get the latest cluster def")
Expect(k8sClient.Get(testCtx.Ctx, client.ObjectKeyFromObject(clusterDefObj), clusterDefObj)).Should(Succeed())
- for i, comp := range clusterObj.Spec.ComponentSpecs {
+ for i, comp := range cluster.Spec.ComponentSpecs {
lorry.SetMockClient(&mockLorryClient{replicas: updatedReplicas, clusterKey: clusterKey, compName: comp.Name}, nil)
By(fmt.Sprintf("H-scale component %s with policy %s", comp.Name, hscalePolicy(comp)))
- horizontalScaleComp(updatedReplicas, &clusterObj.Spec.ComponentSpecs[i], hscalePolicy(comp))
+ horizontalScaleComp(updatedReplicas, &cluster.Spec.ComponentSpecs[i], storageClassName, hscalePolicy(comp))
}
By("Checking cluster status and the number of replicas changed")
Eventually(testapps.GetClusterObservedGeneration(&testCtx, clusterKey)).
- Should(BeEquivalentTo(initialGeneration + len(clusterObj.Spec.ComponentSpecs)))
+ Should(BeEquivalentTo(initialGeneration + len(cluster.Spec.ComponentSpecs)))
}
testHorizontalScale := func(compName, compDefName string, initialReplicas, updatedReplicas int32,
@@ -963,15 +881,14 @@ var _ = Describe("Cluster Controller", func() {
waitForCreatingResourceCompletely(clusterKey, compName)
// REVIEW: this test flow, wait for running phase?
- viper.Set("VOLUMESNAPSHOT", true)
+ testk8s.MockEnableVolumeSnapshot(&testCtx, testk8s.DefaultStorageClassName)
viper.Set(constant.CfgKeyBackupPVCName, "")
- horizontalScale(int(updatedReplicas), dataClonePolicy, compDefName)
+ horizontalScale(int(updatedReplicas), testk8s.DefaultStorageClassName, dataClonePolicy, compDefName)
}
- testVolumeExpansion := func(compName, compDefName string) {
+ testVolumeExpansion := func(compName, compDefName string, storageClass *storagev1.StorageClass) {
var (
- storageClassName = "sc-mock"
replicas = 3
volumeSize = "1Gi"
newVolumeSize = "2Gi"
@@ -980,15 +897,7 @@ var _ = Describe("Cluster Controller", func() {
)
By("Mock a StorageClass which allows resize")
- allowVolumeExpansion := true
- storageClass := &storagev1.StorageClass{
- ObjectMeta: metav1.ObjectMeta{
- Name: storageClassName,
- },
- Provisioner: "kubernetes.io/no-provisioner",
- AllowVolumeExpansion: &allowVolumeExpansion,
- }
- Expect(testCtx.CreateObj(testCtx.Ctx, storageClass)).Should(Succeed())
+ Expect(*storageClass.AllowVolumeExpansion).Should(BeTrue())
By("Creating a cluster with VolumeClaimTemplate")
pvcSpec := testapps.NewPVCSpec(volumeSize)
@@ -1010,18 +919,12 @@ var _ = Describe("Cluster Controller", func() {
Eventually(testapps.GetClusterObservedGeneration(&testCtx, clusterKey)).Should(BeEquivalentTo(1))
By("Checking the replicas")
- var sts *appsv1.StatefulSet
- var rsm *workloads.ReplicatedStateMachine
- if intctrlutil.IsRSMEnabled() {
- rsmList := testk8s.ListAndCheckRSM(&testCtx, clusterKey)
- rsm = &rsmList.Items[0]
- sts = testapps.NewStatefulSetFactory(rsm.Namespace, rsm.Name, clusterObj.Name, compName).
- SetReplicas(*rsm.Spec.Replicas).
- Create(&testCtx).GetObject()
- } else {
- stsList := testk8s.ListAndCheckStatefulSet(&testCtx, clusterKey)
- sts = &stsList.Items[0]
- }
+ rsmList := testk8s.ListAndCheckRSM(&testCtx, clusterKey)
+ rsm := &rsmList.Items[0]
+ sts := testapps.NewStatefulSetFactory(rsm.Namespace, rsm.Name, clusterObj.Name, compName).
+ SetReplicas(*rsm.Spec.Replicas).
+ Create(&testCtx).GetObject()
+
Expect(*sts.Spec.Replicas).Should(BeEquivalentTo(replicas))
By("Mock PVCs in Bound Status")
@@ -1058,11 +961,9 @@ var _ = Describe("Cluster Controller", func() {
case statefulCompDefName, consensusCompDefName:
mockPods = testapps.MockConsensusComponentPods(&testCtx, sts, clusterObj.Name, compName)
}
- if intctrlutil.IsRSMEnabled() {
- Expect(testapps.ChangeObjStatus(&testCtx, rsm, func() {
- testk8s.MockRSMReady(rsm, mockPods...)
- })).ShouldNot(HaveOccurred())
- }
+ Expect(testapps.ChangeObjStatus(&testCtx, rsm, func() {
+ testk8s.MockRSMReady(rsm, mockPods...)
+ })).ShouldNot(HaveOccurred())
Expect(testapps.ChangeObjStatus(&testCtx, sts, func() {
testk8s.MockStatefulSetReady(sts)
})).ShouldNot(HaveOccurred())
@@ -1171,14 +1072,9 @@ var _ = Describe("Cluster Controller", func() {
Eventually(testapps.GetClusterObservedGeneration(&testCtx, clusterKey)).Should(BeEquivalentTo(1))
By("Checking the replicas")
- var numbers int32
- if intctrlutil.IsRSMEnabled() {
- rsmList := testk8s.ListAndCheckRSM(&testCtx, clusterKey)
- numbers = *rsmList.Items[0].Spec.Replicas
- } else {
- stsList := testk8s.ListAndCheckStatefulSet(&testCtx, clusterKey)
- numbers = *stsList.Items[0].Spec.Replicas
- }
+ rsmList := testk8s.ListAndCheckRSM(&testCtx, clusterKey)
+ numbers := *rsmList.Items[0].Spec.Replicas
+
Expect(numbers).Should(BeEquivalentTo(replicas))
By("Mock PVCs in Bound Status")
@@ -1242,13 +1138,8 @@ var _ = Describe("Cluster Controller", func() {
Eventually(testapps.GetClusterObservedGeneration(&testCtx, clusterKey)).Should(BeEquivalentTo(2))
By("Checking PVCs are resized")
- if intctrlutil.IsRSMEnabled() {
- rsmList := testk8s.ListAndCheckRSM(&testCtx, clusterKey)
- numbers = *rsmList.Items[0].Spec.Replicas
- } else {
- stsList := testk8s.ListAndCheckStatefulSet(&testCtx, clusterKey)
- numbers = *stsList.Items[0].Spec.Replicas
- }
+ rsmList = testk8s.ListAndCheckRSM(&testCtx, clusterKey)
+ numbers = *rsmList.Items[0].Spec.Replicas
for i := numbers - 1; i >= 0; i-- {
pvc := &corev1.PersistentVolumeClaim{}
pvcKey := types.NamespacedName{
@@ -1271,13 +1162,8 @@ var _ = Describe("Cluster Controller", func() {
By("Checking PVCs are resized")
Eventually(func(g Gomega) {
- if intctrlutil.IsRSMEnabled() {
- rsmList := testk8s.ListAndCheckRSM(&testCtx, clusterKey)
- numbers = *rsmList.Items[0].Spec.Replicas
- } else {
- stsList := testk8s.ListAndCheckStatefulSet(&testCtx, clusterKey)
- numbers = *stsList.Items[0].Spec.Replicas
- }
+ rsmList = testk8s.ListAndCheckRSM(&testCtx, clusterKey)
+ numbers = *rsmList.Items[0].Spec.Replicas
for i := numbers - 1; i >= 0; i-- {
pvc := &corev1.PersistentVolumeClaim{}
pvcKey := types.NamespacedName{
@@ -1541,34 +1427,8 @@ var _ = Describe("Cluster Controller", func() {
})
}
- mockRoleChangedEvent := func(key types.NamespacedName, sts *appsv1.StatefulSet) []corev1.Event {
- pods, err := components.GetPodListByStatefulSet(ctx, k8sClient, sts)
- Expect(err).To(Succeed())
-
- events := make([]corev1.Event, 0)
- for _, pod := range pods {
- event := corev1.Event{
- ObjectMeta: metav1.ObjectMeta{
- Name: pod.Name + "-event",
- Namespace: testCtx.DefaultNamespace,
- },
- Reason: string(lorryutil.CheckRoleOperation),
- Message: `{"event":"Success","originalRole":"Leader","role":"Follower"}`,
- InvolvedObject: corev1.ObjectReference{
- Name: pod.Name,
- Namespace: testCtx.DefaultNamespace,
- UID: pod.UID,
- FieldPath: constant.ProbeCheckRolePath,
- },
- }
- events = append(events, event)
- }
- events[0].Message = `{"event":"Success","originalRole":"Leader","role":"Leader"}`
- return events
- }
-
getStsPodsName := func(sts *appsv1.StatefulSet) []string {
- pods, err := components.GetPodListByStatefulSet(ctx, k8sClient, sts)
+ pods, err := common.GetPodListByStatefulSet(ctx, k8sClient, sts)
Expect(err).To(Succeed())
names := make([]string, 0)
@@ -1593,29 +1453,20 @@ var _ = Describe("Cluster Controller", func() {
By("Waiting for the cluster controller to create resources completely")
waitForCreatingResourceCompletely(clusterKey, compName)
- var sts *appsv1.StatefulSet
var rsm *workloads.ReplicatedStateMachine
- if intctrlutil.IsRSMEnabled() {
- Eventually(func(g Gomega) {
- rsmList := testk8s.ListAndCheckRSM(&testCtx, clusterKey)
- g.Expect(rsmList.Items).ShouldNot(BeEmpty())
- rsm = &rsmList.Items[0]
- }).Should(Succeed())
- sts = testapps.NewStatefulSetFactory(rsm.Namespace, rsm.Name, clusterKey.Name, compName).
- AddAppComponentLabel(rsm.Labels[constant.KBAppComponentLabelKey]).
- AddAppInstanceLabel(rsm.Labels[constant.AppInstanceLabelKey]).
- SetReplicas(*rsm.Spec.Replicas).Create(&testCtx).GetObject()
- } else {
- Eventually(func(g Gomega) {
- stsList := testk8s.ListAndCheckStatefulSet(&testCtx, clusterKey)
- g.Expect(stsList.Items).ShouldNot(BeEmpty())
- sts = &stsList.Items[0]
- }).Should(Succeed())
- }
+ Eventually(func(g Gomega) {
+ rsmList := testk8s.ListAndCheckRSM(&testCtx, clusterKey)
+ g.Expect(rsmList.Items).ShouldNot(BeEmpty())
+ rsm = &rsmList.Items[0]
+ }).Should(Succeed())
+ sts := testapps.NewStatefulSetFactory(rsm.Namespace, rsm.Name, clusterKey.Name, compName).
+ AddAppComponentLabel(rsm.Labels[constant.KBAppComponentLabelKey]).
+ AddAppInstanceLabel(rsm.Labels[constant.AppInstanceLabelKey]).
+ SetReplicas(*rsm.Spec.Replicas).Create(&testCtx).GetObject()
By("Creating mock pods in StatefulSet, and set controller reference")
pods := mockPodsForTest(clusterObj, replicas)
- for _, pod := range pods {
+ for i, pod := range pods {
Expect(controllerutil.SetControllerReference(sts, &pod, scheme.Scheme)).Should(Succeed())
Expect(testCtx.CreateObj(testCtx.Ctx, &pod)).Should(Succeed())
patch := client.MergeFrom(pod.DeepCopy())
@@ -1624,20 +1475,19 @@ var _ = Describe("Cluster Controller", func() {
Type: corev1.PodReady,
Status: corev1.ConditionTrue,
}}
- // ERROR: the object has been modified; please apply your changes to the latest version and try again
Eventually(k8sClient.Status().Patch(ctx, &pod, patch)).Should(Succeed())
- }
-
- By("Creating mock role changed events")
- // pod.Labels[intctrlutil.RoleLabelKey] will be filled with the role
- events := mockRoleChangedEvent(clusterKey, sts)
- for _, event := range events {
- Expect(testCtx.CreateObj(ctx, &event)).Should(Succeed())
+ role := "follower"
+ if i == 0 {
+ role = "leader"
+ }
+ patch = client.MergeFrom(pod.DeepCopy())
+ pod.Labels[constant.RoleLabelKey] = role
+ Eventually(k8sClient.Patch(ctx, &pod, patch)).Should(Succeed())
}
By("Checking pods' role are changed accordingly")
Eventually(func(g Gomega) {
- pods, err := components.GetPodListByStatefulSet(ctx, k8sClient, sts)
+ pods, err := common.GetPodListByStatefulSet(ctx, k8sClient, sts)
g.Expect(err).ShouldNot(HaveOccurred())
// should have 3 pods
g.Expect(pods).Should(HaveLen(3))
@@ -1656,15 +1506,13 @@ var _ = Describe("Cluster Controller", func() {
g.Expect(followerCount).Should(Equal(2))
}).Should(Succeed())
- if intctrlutil.IsRSMEnabled() {
- // trigger rsm to reconcile as the underlying sts is not created
- Expect(testapps.GetAndChangeObj(&testCtx, client.ObjectKeyFromObject(sts), func(rsm *workloads.ReplicatedStateMachine) {
- rsm.Annotations = map[string]string{"time": time.Now().Format(time.RFC3339)}
- })()).Should(Succeed())
- }
+ // trigger rsm to reconcile as the underlying sts is not created
+ Expect(testapps.GetAndChangeObj(&testCtx, client.ObjectKeyFromObject(sts), func(rsm *workloads.ReplicatedStateMachine) {
+ rsm.Annotations = map[string]string{"time": time.Now().Format(time.RFC3339)}
+ })()).Should(Succeed())
By("Checking pods' annotations")
Eventually(func(g Gomega) {
- pods, err := components.GetPodListByStatefulSet(ctx, k8sClient, sts)
+ pods, err := common.GetPodListByStatefulSet(ctx, k8sClient, sts)
g.Expect(err).ShouldNot(HaveOccurred())
g.Expect(pods).Should(HaveLen(int(*sts.Spec.Replicas)))
for _, pod := range pods {
@@ -1672,19 +1520,18 @@ var _ = Describe("Cluster Controller", func() {
g.Expect(pod.Annotations[constant.ComponentReplicasAnnotationKey]).Should(Equal(strconv.Itoa(int(*sts.Spec.Replicas))))
}
}).Should(Succeed())
- if intctrlutil.IsRSMEnabled() {
- rsmPatch := client.MergeFrom(rsm.DeepCopy())
- By("Updating RSM's status")
- rsm.Status.UpdateRevision = "mock-version"
- pods, err := components.GetPodListByStatefulSet(ctx, k8sClient, sts)
- Expect(err).Should(BeNil())
- var podList []*corev1.Pod
- for i := range pods {
- podList = append(podList, &pods[i])
- }
- testk8s.MockRSMReady(rsm, podList...)
- Expect(k8sClient.Status().Patch(ctx, rsm, rsmPatch)).Should(Succeed())
- }
+ rsmPatch := client.MergeFrom(rsm.DeepCopy())
+ By("Updating RSM's status")
+ rsm.Status.UpdateRevision = "mock-version"
+ pods, err := common.GetPodListByStatefulSet(ctx, k8sClient, sts)
+ Expect(err).Should(BeNil())
+ var podList []*corev1.Pod
+ for i := range pods {
+ podList = append(podList, &pods[i])
+ }
+ testk8s.MockRSMReady(rsm, podList...)
+ Expect(k8sClient.Status().Patch(ctx, rsm, rsmPatch)).Should(Succeed())
+
stsPatch := client.MergeFrom(sts.DeepCopy())
By("Updating StatefulSet's status")
sts.Status.UpdateRevision = "mock-version"
@@ -1720,7 +1567,7 @@ var _ = Describe("Cluster Controller", func() {
testBackupError := func(compName, compDefName string) {
initialReplicas := int32(1)
updatedReplicas := int32(3)
- viper.Set("VOLUMESNAPSHOT", true)
+ testk8s.MockEnableVolumeSnapshot(&testCtx, testk8s.DefaultStorageClassName)
By("Set HorizontalScalePolicy")
Expect(testapps.GetAndChangeObj(&testCtx, client.ObjectKeyFromObject(clusterDefObj),
@@ -1749,6 +1596,11 @@ var _ = Describe("Cluster Controller", func() {
waitForCreatingResourceCompletely(clusterKey, compName)
Eventually(testapps.GetClusterObservedGeneration(&testCtx, clusterKey)).Should(BeEquivalentTo(1))
+ By("Create and Mock PVCs status to bound")
+ for _, comp := range clusterObj.Spec.ComponentSpecs {
+ mockComponentPVCsAndBound(&comp, int(comp.Replicas), true, testk8s.DefaultStorageClassName)
+ }
+
By(fmt.Sprintf("Changing replicas to %d", updatedReplicas))
changeCompReplicas(clusterKey, updatedReplicas, &clusterObj.Spec.ComponentSpecs[0])
Eventually(testapps.GetClusterObservedGeneration(&testCtx, clusterKey)).Should(BeEquivalentTo(2))
@@ -1762,32 +1614,27 @@ var _ = Describe("Cluster Controller", func() {
ml, client.InNamespace(clusterKey.Namespace))).Should(HaveLen(1))
By("Mocking backup status to failed")
- backupList := dataprotectionv1alpha1.BackupList{}
+ backupList := dpv1alpha1.BackupList{}
Expect(testCtx.Cli.List(testCtx.Ctx, &backupList, ml)).Should(Succeed())
backupKey := types.NamespacedName{
Namespace: backupList.Items[0].Namespace,
Name: backupList.Items[0].Name,
}
- Expect(testapps.GetAndChangeObjStatus(&testCtx, backupKey, func(backup *dataprotectionv1alpha1.Backup) {
- backup.Status.Phase = dataprotectionv1alpha1.BackupFailed
+ Expect(testapps.GetAndChangeObjStatus(&testCtx, backupKey, func(backup *dpv1alpha1.Backup) {
+ backup.Status.Phase = dpv1alpha1.BackupPhaseFailed
})()).Should(Succeed())
By("Checking cluster status failed with backup error")
Eventually(testapps.CheckObj(&testCtx, clusterKey, func(g Gomega, cluster *appsv1alpha1.Cluster) {
- g.Expect(viper.GetBool("VOLUMESNAPSHOT")).Should(BeTrue())
+ g.Expect(testk8s.IsMockVolumeSnapshotEnabled(&testCtx, testk8s.DefaultStorageClassName)).Should(BeTrue())
g.Expect(cluster.Status.Conditions).ShouldNot(BeEmpty())
var err error
for _, cond := range cluster.Status.Conditions {
if strings.Contains(cond.Message, "backup for horizontalScaling failed") {
- g.Expect(cond.Message).Should(ContainSubstring("backup for horizontalScaling failed"))
err = errors.New("has backup error")
break
}
}
- if err == nil {
- // this expectation is intended to print all cluster.Status.Conditions
- g.Expect(cluster.Status.Conditions).Should(BeEmpty())
- }
g.Expect(err).Should(HaveOccurred())
})).Should(Succeed())
@@ -1951,96 +1798,6 @@ var _ = Describe("Cluster Controller", func() {
factory.SetReplicas(3)
}, true)
- By("Check deployment workload has been created")
- Eventually(testapps.List(&testCtx, generics.DeploymentSignature,
- client.MatchingLabels{
- constant.AppInstanceLabelKey: clusterKey.Name,
- }, client.InNamespace(clusterKey.Namespace))).ShouldNot(HaveLen(0))
-
- stsList := testk8s.ListAndCheckStatefulSet(&testCtx, clusterKey)
-
- By("Check statefulset pod's volumes")
- for _, sts := range stsList.Items {
- podSpec := sts.Spec.Template
- volumeNames := map[string]struct{}{}
- for _, v := range podSpec.Spec.Volumes {
- volumeNames[v.Name] = struct{}{}
- }
-
- for _, cc := range [][]corev1.Container{
- podSpec.Spec.Containers,
- podSpec.Spec.InitContainers,
- } {
- for _, c := range cc {
- for _, vm := range c.VolumeMounts {
- _, ok := volumeNames[vm.Name]
- Expect(ok).Should(BeTrue())
- }
- }
- }
- }
-
- By("Check associated PDB has been created")
- Eventually(testapps.List(&testCtx, generics.PodDisruptionBudgetSignature,
- client.MatchingLabels{
- constant.AppInstanceLabelKey: clusterKey.Name,
- }, client.InNamespace(clusterKey.Namespace))).ShouldNot(BeEmpty())
-
- podSpec := stsList.Items[0].Spec.Template.Spec
- By("Checking created sts pods template with built-in toleration")
- Expect(podSpec.Tolerations).Should(HaveLen(1))
- Expect(podSpec.Tolerations[0].Key).To(Equal(testDataPlaneTolerationKey))
-
- By("Checking created sts pods template with built-in Affinity")
- Expect(podSpec.Affinity.PodAntiAffinity == nil && podSpec.Affinity.PodAffinity == nil).Should(BeTrue())
- Expect(podSpec.Affinity.NodeAffinity).ShouldNot(BeNil())
- Expect(podSpec.Affinity.NodeAffinity.PreferredDuringSchedulingIgnoredDuringExecution[0].Preference.MatchExpressions[0].Key).To(
- Equal(testDataPlaneNodeAffinityKey))
-
- By("Checking created sts pods template without TopologySpreadConstraints")
- Expect(podSpec.TopologySpreadConstraints).Should(BeEmpty())
-
- By("Check should create env configmap")
- Eventually(func(g Gomega) {
- cmList := &corev1.ConfigMapList{}
- Expect(k8sClient.List(testCtx.Ctx, cmList, client.MatchingLabels{
- constant.AppInstanceLabelKey: clusterKey.Name,
- constant.AppConfigTypeLabelKey: "kubeblocks-env",
- }, client.InNamespace(clusterKey.Namespace))).Should(Succeed())
- Expect(cmList.Items).ShouldNot(BeEmpty())
- Expect(cmList.Items).Should(HaveLen(len(compNameNDef)))
- }).Should(Succeed())
-
- By("Checking stateless services")
- statelessExpectServices := map[string]ExpectService{
- // TODO: fix me later, proxy should not have internal headless service
- testapps.ServiceHeadlessName: {svcType: corev1.ServiceTypeClusterIP, headless: true},
- testapps.ServiceDefaultName: {svcType: corev1.ServiceTypeClusterIP, headless: false},
- }
- Eventually(func(g Gomega) {
- validateCompSvcList(g, statelessCompName, statelessCompDefName, statelessExpectServices)
- }).Should(Succeed())
-
- By("Checking stateful types services")
- for compName, compNameNDef := range compNameNDef {
- if compName == statelessCompName {
- continue
- }
- consensusExpectServices := map[string]ExpectService{
- testapps.ServiceHeadlessName: {svcType: corev1.ServiceTypeClusterIP, headless: true},
- testapps.ServiceDefaultName: {svcType: corev1.ServiceTypeClusterIP, headless: false},
- }
- Eventually(func(g Gomega) {
- validateCompSvcList(g, compName, compNameNDef, consensusExpectServices)
- }).Should(Succeed())
- }
- }
-
- checkAllResourcesCreatedWithRSMEnabled := func(compNameNDef map[string]string) {
- createNWaitClusterObj(compNameNDef, func(compName string, factory *testapps.MockClusterFactory) {
- factory.SetReplicas(3)
- }, true)
-
By("Check stateless workload has been created")
Eventually(testapps.List(&testCtx, generics.RSMSignature,
client.MatchingLabels{
@@ -2091,19 +1848,6 @@ var _ = Describe("Cluster Controller", func() {
By("Checking created rsm pods template without TopologySpreadConstraints")
Expect(podSpec.TopologySpreadConstraints).Should(BeEmpty())
- if !intctrlutil.IsRSMEnabled() {
- By("Check should create env configmap")
- Eventually(func(g Gomega) {
- cmList := &corev1.ConfigMapList{}
- g.Expect(k8sClient.List(testCtx.Ctx, cmList, client.MatchingLabels{
- constant.AppInstanceLabelKey: clusterKey.Name,
- constant.AppConfigTypeLabelKey: "kubeblocks-env",
- }, client.InNamespace(clusterKey.Namespace))).Should(Succeed())
- g.Expect(cmList.Items).ShouldNot(BeEmpty())
- g.Expect(cmList.Items).Should(HaveLen(len(compNameNDef)))
- }).Should(Succeed())
- }
-
By("Checking stateless services")
statelessExpectServices := map[string]ExpectService{
// TODO: fix me later, proxy should not have internal headless service
@@ -2151,7 +1895,7 @@ var _ = Describe("Cluster Controller", func() {
// statefulCompDefName not in componentDefsWithHScalePolicy, for nil backup policy test
// REVIEW:
// 1. this test flow, wait for running phase?
- horizontalScale(int(updatedReplicas), policyType, consensusCompDefName, replicationCompDefName)
+ horizontalScale(int(updatedReplicas), testk8s.DefaultStorageClassName, policyType, consensusCompDefName, replicationCompDefName)
}
It("should create all sub-resources successfully, with terminationPolicy=Halt lifecycle", func() {
@@ -2161,24 +1905,13 @@ var _ = Describe("Cluster Controller", func() {
statefulCompName: statefulCompDefName,
replicationCompName: replicationCompDefName,
}
- if intctrlutil.IsRSMEnabled() {
- checkAllResourcesCreatedWithRSMEnabled(compNameNDef)
- } else {
- checkAllResourcesCreated(compNameNDef)
- }
+ checkAllResourcesCreated(compNameNDef)
By("Mocking components' PVCs to bound")
var items []client.Object
- if intctrlutil.IsRSMEnabled() {
- rsmList := testk8s.ListAndCheckRSM(&testCtx, clusterKey)
- for i := range rsmList.Items {
- items = append(items, &rsmList.Items[i])
- }
- } else {
- stsList := testk8s.ListAndCheckStatefulSet(&testCtx, clusterKey)
- for i := range stsList.Items {
- items = append(items, &stsList.Items[i])
- }
+ rsmList := testk8s.ListAndCheckRSM(&testCtx, clusterKey)
+ for i := range rsmList.Items {
+ items = append(items, &rsmList.Items[i])
}
for _, item := range items {
compName, ok := item.GetLabels()[constant.KBAppComponentLabelKey]
@@ -2189,7 +1922,7 @@ var _ = Describe("Cluster Controller", func() {
Namespace: clusterKey.Namespace,
Name: getPVCName(testapps.DataVolumeName, compName, i),
}
- createPVC(clusterKey.Name, pvcKey.Name, compName)
+ createPVC(clusterKey.Name, pvcKey.Name, compName, "", "")
Eventually(testapps.CheckObjExists(&testCtx, pvcKey, &corev1.PersistentVolumeClaim{}, true)).Should(Succeed())
Expect(testapps.GetAndChangeObjStatus(&testCtx, pvcKey, func(pvc *corev1.PersistentVolumeClaim) {
pvc.Status.Phase = corev1.ClaimBound
@@ -2197,7 +1930,7 @@ var _ = Describe("Cluster Controller", func() {
}
}
- By("delete the cluster and should preserved PVC,Secret,CM resources")
+ By("delete the cluster and should be preserved PVC,Secret,CM resources")
deleteCluster := func(termPolicy appsv1alpha1.TerminationPolicyType) {
// TODO: would be better that cluster is created with terminationPolicy=Halt instead of
// reassign the value after created
@@ -2246,27 +1979,17 @@ var _ = Describe("Cluster Controller", func() {
for _, secret := range secretList.Items {
checkObject(&secret)
}
- if !intctrlutil.IsRSMEnabled() {
- By("check configmap resources preserved")
- Expect(cmList.Items).ShouldNot(BeEmpty())
- for _, cm := range cmList.Items {
- checkObject(&cm)
- }
- }
}
return pvcList, secretList, cmList
}
- initPVCList, initSecretList, initCMList := checkPreservedObjects(clusterObj.UID)
+ initPVCList, initSecretList, _ := checkPreservedObjects(clusterObj.UID)
By("create recovering cluster")
lastClusterUID := clusterObj.UID
- if intctrlutil.IsRSMEnabled() {
- checkAllResourcesCreatedWithRSMEnabled(compNameNDef)
- } else {
- checkAllResourcesCreated(compNameNDef)
- }
+ checkAllResourcesCreated(compNameNDef)
+
Expect(clusterObj.UID).ShouldNot(Equal(lastClusterUID))
- lastPVCList, lastSecretList, lastCMList := checkPreservedObjects("")
+ lastPVCList, lastSecretList, _ := checkPreservedObjects("")
Expect(outOfOrderEqualFunc(initPVCList.Items, lastPVCList.Items, func(i corev1.PersistentVolumeClaim, j corev1.PersistentVolumeClaim) bool {
return i.UID == j.UID
@@ -2274,25 +1997,20 @@ var _ = Describe("Cluster Controller", func() {
Expect(outOfOrderEqualFunc(initSecretList.Items, lastSecretList.Items, func(i corev1.Secret, j corev1.Secret) bool {
return i.UID == j.UID
})).Should(BeTrue())
- if !intctrlutil.IsRSMEnabled() {
- Expect(outOfOrderEqualFunc(initCMList.Items, lastCMList.Items, func(i corev1.ConfigMap, j corev1.ConfigMap) bool {
- return i.UID == j.UID
- })).Should(BeTrue())
- }
- By("delete the cluster and should preserved PVC,Secret,CM resources but result updated the new last applied cluster UID")
+ By("delete the cluster and should be preserved PVC,Secret,CM resources but result updated the new last applied cluster UID")
deleteCluster(appsv1alpha1.Halt)
checkPreservedObjects(clusterObj.UID)
})
It("should successfully h-scale with multiple components", func() {
- viper.Set("VOLUMESNAPSHOT", true)
+ testk8s.MockEnableVolumeSnapshot(&testCtx, testk8s.DefaultStorageClassName)
viper.Set(constant.CfgKeyBackupPVCName, "")
testMultiCompHScale(appsv1alpha1.HScaleDataClonePolicyCloneVolume)
})
It("should successfully h-scale with multiple components by backup tool", func() {
- viper.Set("VOLUMESNAPSHOT", false)
+ testk8s.MockDisableVolumeSnapshot(&testCtx, testk8s.DefaultStorageClassName)
viper.Set(constant.CfgKeyBackupPVCName, "test-backup-pvc")
testMultiCompHScale(appsv1alpha1.HScaleDataClonePolicyCloneVolume)
})
@@ -2300,9 +2018,11 @@ var _ = Describe("Cluster Controller", func() {
When("creating cluster with backup configuration", func() {
const (
- compName = statefulCompName
- compDefName = statefulCompDefName
- backupRepoName = "test-backup-repo"
+ compName = statefulCompName
+ compDefName = statefulCompDefName
+ backupRepoName = "test-backup-repo"
+ backupMethodName = "test-backup-method"
+ volumeSnapshotBackupMethodName = "test-vs-backup-method"
)
BeforeEach(func() {
cleanEnv()
@@ -2337,8 +2057,8 @@ var _ = Describe("Cluster Controller", func() {
int64Ptr = func(in int64) *int64 {
return &in
}
- strPtr = func(s string) *string {
- return &s
+ retention = func(s string) dpv1alpha1.RetentionPeriod {
+ return dpv1alpha1.RetentionPeriod(s)
}
)
@@ -2350,8 +2070,8 @@ var _ = Describe("Cluster Controller", func() {
desc: "backup with snapshot method",
backup: &appsv1alpha1.ClusterBackup{
Enabled: &boolTrue,
- RetentionPeriod: strPtr("1d"),
- Method: dataprotectionv1alpha1.BackupMethodSnapshot,
+ RetentionPeriod: retention("1d"),
+ Method: vsBackupMethodName,
CronExpression: "*/1 * * * *",
StartingDeadlineMinutes: int64Ptr(int64(10)),
PITREnabled: &boolTrue,
@@ -2362,8 +2082,8 @@ var _ = Describe("Cluster Controller", func() {
desc: "disable backup",
backup: &appsv1alpha1.ClusterBackup{
Enabled: &boolFalse,
- RetentionPeriod: strPtr("1d"),
- Method: dataprotectionv1alpha1.BackupMethodSnapshot,
+ RetentionPeriod: retention("1d"),
+ Method: vsBackupMethodName,
CronExpression: "*/1 * * * *",
StartingDeadlineMinutes: int64Ptr(int64(10)),
PITREnabled: &boolTrue,
@@ -2371,11 +2091,11 @@ var _ = Describe("Cluster Controller", func() {
},
},
{
- desc: "backup with backup tool method",
+ desc: "backup with backup tool",
backup: &appsv1alpha1.ClusterBackup{
Enabled: &boolTrue,
- RetentionPeriod: strPtr("2d"),
- Method: dataprotectionv1alpha1.BackupMethodBackupTool,
+ RetentionPeriod: retention("2d"),
+ Method: backupMethodName,
CronExpression: "*/1 * * * *",
StartingDeadlineMinutes: int64Ptr(int64(10)),
RepoName: backupRepoName,
@@ -2392,47 +2112,37 @@ var _ = Describe("Cluster Controller", func() {
By(t.desc)
backup := t.backup
createClusterWithBackup(backup)
- checkSchedulePolicy := func(g Gomega, sp *dataprotectionv1alpha1.SchedulePolicy) {
- g.Expect(sp).ShouldNot(BeNil())
- g.Expect(sp.Enable).Should(BeEquivalentTo(*backup.Enabled))
- g.Expect(sp.CronExpression).Should(Equal(backup.CronExpression))
- }
- checkPolicy := func(g Gomega, p *dataprotectionv1alpha1.BackupPolicy) {
- schedule := p.Spec.Schedule
- switch backup.Method {
- case dataprotectionv1alpha1.BackupMethodSnapshot:
- checkSchedulePolicy(g, schedule.Snapshot)
- case dataprotectionv1alpha1.BackupMethodBackupTool:
- checkSchedulePolicy(g, schedule.Datafile)
+
+ checkSchedule := func(g Gomega, schedule *dpv1alpha1.BackupSchedule) {
+ var policy *dpv1alpha1.SchedulePolicy
+ for i, s := range schedule.Spec.Schedules {
+ if s.BackupMethod == backup.Method {
+ Expect(*s.Enabled).Should(BeEquivalentTo(*backup.Enabled))
+ policy = &schedule.Spec.Schedules[i]
+ }
}
- g.Expect(schedule.Logfile.Enable).Should(BeEquivalentTo(*backup.PITREnabled))
- g.Expect(*p.Spec.Logfile.BackupRepoName).Should(BeEquivalentTo(backup.RepoName))
- g.Expect(schedule.StartingDeadlineMinutes).Should(Equal(backup.StartingDeadlineMinutes))
- }
- checkPolicyDisabled := func(g Gomega, p *dataprotectionv1alpha1.BackupPolicy) {
- schedule := p.Spec.Schedule
- switch backup.Method {
- case dataprotectionv1alpha1.BackupMethodSnapshot:
- g.Expect(schedule.Snapshot.Enable).Should(BeFalse())
- case dataprotectionv1alpha1.BackupMethodBackupTool:
- g.Expect(schedule.Datafile.Enable).Should(BeFalse())
+ if backup.Enabled != nil && *backup.Enabled {
+ Expect(policy).ShouldNot(BeNil())
+ Expect(policy.RetentionPeriod).Should(BeEquivalentTo(backup.RetentionPeriod))
+ Expect(policy.CronExpression).Should(BeEquivalentTo(backup.CronExpression))
}
}
- policyName := DeriveBackupPolicyName(clusterKey.Name, compDefName, "")
- Eventually(testapps.CheckObj(&testCtx, client.ObjectKey{Name: policyName, Namespace: clusterKey.Namespace},
- func(g Gomega, policy *dataprotectionv1alpha1.BackupPolicy) {
- if backup == nil {
- // if cluster.Spec.Backup is nil, will use the default backup policy
- g.Expect(policy).ShouldNot(BeNil())
- g.Expect(policy.Spec.Schedule).ShouldNot(BeNil())
- g.Expect(policy.Spec.Schedule.Snapshot).ShouldNot(BeNil())
- g.Expect(policy.Spec.Schedule.Snapshot.Enable).Should(BeFalse())
- } else if boolValue(backup.Enabled) {
- checkPolicy(g, policy)
- } else {
- checkPolicyDisabled(g, policy)
- }
- })).Should(Succeed())
+
+ By("checking backup policy exists")
+ backupPolicyName := generateBackupPolicyName(clusterKey.Name, compDefName, "")
+ backupPolicyKey := client.ObjectKey{Name: backupPolicyName, Namespace: clusterKey.Namespace}
+ backupPolicy := &dpv1alpha1.BackupPolicy{}
+ Eventually(testapps.CheckObjExists(&testCtx, backupPolicyKey, backupPolicy, true)).Should(Succeed())
+
+ By("checking backup schedule")
+ backupScheduleName := generateBackupScheduleName(clusterKey.Name, compDefName, "")
+ backupScheduleKey := client.ObjectKey{Name: backupScheduleName, Namespace: clusterKey.Namespace}
+ if backup == nil {
+ Eventually(testapps.CheckObjExists(&testCtx, backupScheduleKey,
+ &dpv1alpha1.BackupSchedule{}, false)).Should(Succeed())
+ continue
+ }
+ Eventually(testapps.CheckObj(&testCtx, backupScheduleKey, checkSchedule)).Should(Succeed())
}
})
})
@@ -2513,6 +2223,10 @@ var _ = Describe("Cluster Controller", func() {
})
When("creating cluster with stateful workloadTypes (being Stateful|Consensus|Replication) component", func() {
+ var (
+ mockStorageClass *storagev1.StorageClass
+ )
+
compNameNDef := map[string]string{
statefulCompName: statefulCompDefName,
consensusCompName: consensusCompDefName,
@@ -2522,12 +2236,13 @@ var _ = Describe("Cluster Controller", func() {
BeforeEach(func() {
createAllWorkloadTypesClusterDef()
createBackupPolicyTpl(clusterDefObj)
+ mockStorageClass = testk8s.CreateMockStorageClass(&testCtx, testk8s.DefaultStorageClassName)
})
for compName, compDefName := range compNameNDef {
Context(fmt.Sprintf("[comp: %s] volume expansion", compName), func() {
It("should update PVC request storage size accordingly", func() {
- testVolumeExpansion(compName, compDefName)
+ testVolumeExpansion(compName, compDefName, mockStorageClass)
})
It("should be able to recover if volume expansion fails", func() {
@@ -2563,15 +2278,15 @@ var _ = Describe("Cluster Controller", func() {
Context(fmt.Sprintf("[comp: %s] scale-out after volume expansion", compName), func() {
It("scale-out with data clone policy", func() {
- testVolumeExpansion(compName, compDefName)
- viper.Set("VOLUMESNAPSHOT", true)
+ testVolumeExpansion(compName, compDefName, mockStorageClass)
+ testk8s.MockEnableVolumeSnapshot(&testCtx, mockStorageClass.Name)
viper.Set(constant.CfgKeyBackupPVCName, "")
- horizontalScale(5, appsv1alpha1.HScaleDataClonePolicyCloneVolume, compDefName)
+ horizontalScale(5, mockStorageClass.Name, appsv1alpha1.HScaleDataClonePolicyCloneVolume, compDefName)
})
It("scale-out without data clone policy", func() {
- testVolumeExpansion(compName, compDefName)
- horizontalScale(5, "", compDefName)
+ testVolumeExpansion(compName, compDefName, mockStorageClass)
+ horizontalScale(5, mockStorageClass.Name, "", compDefName)
})
})
}
@@ -2596,102 +2311,77 @@ var _ = Describe("Cluster Controller", func() {
By("mock backuptool object")
backupPolicyName := "test-backup-policy"
backupName := "test-backup"
- backupTool := testapps.CreateCustomizedObj(&testCtx, "backup/backuptool.yaml",
- &dataprotectionv1alpha1.BackupTool{}, testapps.RandomizedObjName())
+ _ = testapps.CreateCustomizedObj(&testCtx, "backup/actionset.yaml",
+ &dpv1alpha1.ActionSet{}, testapps.RandomizedObjName())
By("creating backup")
- backup := testapps.NewBackupFactory(testCtx.DefaultNamespace, backupName).
+ backup := testdp.NewBackupFactory(testCtx.DefaultNamespace, backupName).
SetBackupPolicyName(backupPolicyName).
- SetBackupType(dataprotectionv1alpha1.BackupTypeDataFile).
+ SetBackupMethod(testdp.BackupMethodName).
Create(&testCtx).GetObject()
By("mocking backup status completed, we don't need backup reconcile here")
- Eventually(testapps.GetAndChangeObjStatus(&testCtx, client.ObjectKeyFromObject(backup), func(backup *dataprotectionv1alpha1.Backup) {
- backup.Status.BackupToolName = backupTool.Name
+ Eventually(testapps.GetAndChangeObjStatus(&testCtx, client.ObjectKeyFromObject(backup), func(backup *dpv1alpha1.Backup) {
backup.Status.PersistentVolumeClaimName = "backup-pvc"
- backup.Status.Phase = dataprotectionv1alpha1.BackupCompleted
+ backup.Status.Phase = dpv1alpha1.BackupPhaseCompleted
+ testdp.MockBackupStatusMethod(backup, testapps.DataVolumeName)
})).Should(Succeed())
- By("checking backup status completed")
- Eventually(testapps.CheckObj(&testCtx, client.ObjectKeyFromObject(backup),
- func(g Gomega, tmpBackup *dataprotectionv1alpha1.Backup) {
- g.Expect(tmpBackup.Status.Phase).Should(Equal(dataprotectionv1alpha1.BackupCompleted))
- })).Should(Succeed())
-
By("creating cluster with backup")
- restoreFromBackup := fmt.Sprintf(`{"%s":"%s"}`, compName, backupName)
+ restoreFromBackup := fmt.Sprintf(`{"%s":{"name":"%s"}}`, compName, backupName)
pvcSpec := testapps.NewPVCSpec("1Gi")
+ replicas := 3
clusterObj = testapps.NewClusterFactory(testCtx.DefaultNamespace, clusterName,
clusterDefObj.Name, clusterVersionObj.Name).WithRandomName().
AddComponent(compName, compDefName).
- SetReplicas(3).
+ SetReplicas(int32(replicas)).
AddVolumeClaimTemplate(testapps.DataVolumeName, pvcSpec).
- AddAnnotations(constant.RestoreFromBackUpAnnotationKey, restoreFromBackup).Create(&testCtx).GetObject()
+ AddAnnotations(constant.RestoreFromBackupAnnotationKey, restoreFromBackup).Create(&testCtx).GetObject()
clusterKey = client.ObjectKeyFromObject(clusterObj)
- By("mocking restore job completed")
- patchK8sJobStatus := func(key types.NamespacedName, jobStatus batchv1.JobConditionType) {
- Eventually(testapps.GetAndChangeObjStatus(&testCtx, key, func(fetched *batchv1.Job) {
- jobCondition := batchv1.JobCondition{Type: jobStatus}
- fetched.Status.Conditions = append(fetched.Status.Conditions, jobCondition)
- })).Should(Succeed())
- }
- for i := 0; i < 3; i++ {
- restoreJobKey := client.ObjectKey{
- Name: fmt.Sprintf("base-%s-%s-%s-%d", testapps.DataVolumeName, clusterObj.Name, compName, i),
- Namespace: clusterKey.Namespace,
- }
- patchK8sJobStatus(restoreJobKey, batchv1.JobComplete)
+ // mock pvcs have restored
+ mockComponentPVCsAndBound(clusterObj.Spec.GetComponentByName(compName), replicas, true, testk8s.DefaultStorageClassName)
+ By("wait for restore created")
+ ml := client.MatchingLabels{
+ constant.AppInstanceLabelKey: clusterKey.Name,
+ constant.KBAppComponentLabelKey: compName,
}
+ Eventually(testapps.List(&testCtx, generics.RestoreSignature,
+ ml, client.InNamespace(clusterKey.Namespace))).Should(HaveLen(1))
+
+ By("Mocking restore phase to Completed")
+ // mock prepareData restore completed
+ mockRestoreCompleted(ml)
By("Waiting for the cluster controller to create resources completely")
waitForCreatingResourceCompletely(clusterKey, compName)
- if intctrlutil.IsRSMEnabled() {
- rsmList := testk8s.ListAndCheckRSM(&testCtx, clusterKey)
- rsm := rsmList.Items[0]
- sts := testapps.NewStatefulSetFactory(rsm.Namespace, rsm.Name, clusterKey.Name, compName).
- SetReplicas(*rsm.Spec.Replicas).
- Create(&testCtx).GetObject()
- By("mock pod/sts are available and wait for component enter running phase")
- mockPods := testapps.MockConsensusComponentPods(&testCtx, sts, clusterObj.Name, compName)
- Expect(testapps.ChangeObjStatus(&testCtx, sts, func() {
- testk8s.MockStatefulSetReady(sts)
- })).ShouldNot(HaveOccurred())
- Expect(testapps.ChangeObjStatus(&testCtx, &rsm, func() {
- testk8s.MockRSMReady(&rsm, mockPods...)
- })).ShouldNot(HaveOccurred())
- Eventually(testapps.GetClusterComponentPhase(&testCtx, clusterKey, compName)).Should(Equal(appsv1alpha1.RunningClusterCompPhase))
-
- By("the restore container has been removed from init containers")
- Eventually(testapps.CheckObj(&testCtx, client.ObjectKeyFromObject(&rsm), func(g Gomega, tmpRSM *workloads.ReplicatedStateMachine) {
- g.Expect(tmpRSM.Spec.Template.Spec.InitContainers).Should(BeEmpty())
- })).Should(Succeed())
- } else {
- stsList := testk8s.ListAndCheckStatefulSet(&testCtx, clusterKey)
- sts := stsList.Items[0]
- By("mock pod/sts are available and wait for component enter running phase")
- testapps.MockConsensusComponentPods(&testCtx, &sts, clusterObj.Name, compName)
- Expect(testapps.ChangeObjStatus(&testCtx, &sts, func() {
- testk8s.MockStatefulSetReady(&sts)
- })).ShouldNot(HaveOccurred())
- Eventually(testapps.GetClusterComponentPhase(&testCtx, clusterKey, compName)).Should(Equal(appsv1alpha1.RunningClusterCompPhase))
-
- By("the restore container has been removed from init containers")
- Eventually(testapps.CheckObj(&testCtx, client.ObjectKeyFromObject(&sts), func(g Gomega, tmpSts *appsv1.StatefulSet) {
- g.Expect(tmpSts.Spec.Template.Spec.InitContainers).Should(BeEmpty())
- })).Should(Succeed())
- }
+ rsmList := testk8s.ListAndCheckRSM(&testCtx, clusterKey)
+ rsm := rsmList.Items[0]
+ sts := testapps.NewStatefulSetFactory(rsm.Namespace, rsm.Name, clusterKey.Name, compName).
+ SetReplicas(*rsm.Spec.Replicas).
+ Create(&testCtx).GetObject()
+ By("mock pod/sts are available and wait for component enter running phase")
+ mockPods := testapps.MockConsensusComponentPods(&testCtx, sts, clusterObj.Name, compName)
+ Expect(testapps.ChangeObjStatus(&testCtx, sts, func() {
+ testk8s.MockStatefulSetReady(sts)
+ })).ShouldNot(HaveOccurred())
+ Expect(testapps.ChangeObjStatus(&testCtx, &rsm, func() {
+ testk8s.MockRSMReady(&rsm, mockPods...)
+ })).ShouldNot(HaveOccurred())
+ Eventually(testapps.GetClusterComponentPhase(&testCtx, clusterKey, compName)).Should(Equal(appsv1alpha1.RunningClusterCompPhase))
+
+ By("the restore container has been removed from init containers")
+ Eventually(testapps.CheckObj(&testCtx, client.ObjectKeyFromObject(&rsm), func(g Gomega, tmpRSM *workloads.ReplicatedStateMachine) {
+ g.Expect(tmpRSM.Spec.Template.Spec.InitContainers).Should(BeEmpty())
+ })).Should(Succeed())
By("clean up annotations after cluster running")
- Expect(testapps.GetAndChangeObjStatus(&testCtx, clusterKey, func(tmpCluster *appsv1alpha1.Cluster) {
- compStatus := tmpCluster.Status.Components[compName]
- compStatus.Phase = appsv1alpha1.RunningClusterCompPhase
- tmpCluster.Status.Components[compName] = compStatus
- })()).Should(Succeed())
Eventually(testapps.CheckObj(&testCtx, clusterKey, func(g Gomega, tmpCluster *appsv1alpha1.Cluster) {
g.Expect(tmpCluster.Status.Phase).Should(Equal(appsv1alpha1.RunningClusterPhase))
- g.Expect(tmpCluster.Annotations[constant.RestoreFromBackUpAnnotationKey]).Should(BeEmpty())
+ // mock postReady restore completed
+ mockRestoreCompleted(ml)
+ g.Expect(tmpCluster.Annotations[constant.RestoreFromBackupAnnotationKey]).Should(BeEmpty())
})).Should(Succeed())
})
})
@@ -2723,31 +2413,17 @@ var _ = Describe("Cluster Controller", func() {
waitForCreatingResourceCompletely(clusterKey, compDefName)
By("Checking statefulSet number")
- var sts *appsv1.StatefulSet
- if intctrlutil.IsRSMEnabled() {
- rsmList := testk8s.ListAndCheckRSMItemsCount(&testCtx, clusterKey, 1)
- rsm := &rsmList.Items[0]
- sts = testapps.NewStatefulSetFactory(rsm.Namespace, rsm.Name, clusterKey.Name, compName).
- SetReplicas(*rsm.Spec.Replicas).Create(&testCtx).GetObject()
- mockPods := testapps.MockReplicationComponentPods(nil, testCtx, sts, clusterObj.Name, compDefName, nil)
- Expect(testapps.ChangeObjStatus(&testCtx, sts, func() {
- testk8s.MockStatefulSetReady(sts)
- })).ShouldNot(HaveOccurred())
- Expect(testapps.ChangeObjStatus(&testCtx, rsm, func() {
- testk8s.MockRSMReady(rsm, mockPods...)
- })).ShouldNot(HaveOccurred())
- } else {
- stsList := testk8s.ListAndCheckStatefulSetItemsCount(&testCtx, clusterKey, 1)
- sts = &stsList.Items[0]
- Expect(testapps.ChangeObjStatus(&testCtx, sts, func() {
- testk8s.MockStatefulSetReady(sts)
- })).ShouldNot(HaveOccurred())
- for i := int32(0); i < *sts.Spec.Replicas; i++ {
- podName := fmt.Sprintf("%s-%d", sts.Name, i)
- testapps.MockReplicationComponentPod(nil, testCtx, sts, clusterObj.Name,
- compDefName, podName, components.DefaultRole(i))
- }
- }
+ rsmList := testk8s.ListAndCheckRSMItemsCount(&testCtx, clusterKey, 1)
+ rsm := &rsmList.Items[0]
+ sts := testapps.NewStatefulSetFactory(rsm.Namespace, rsm.Name, clusterKey.Name, compName).
+ SetReplicas(*rsm.Spec.Replicas).Create(&testCtx).GetObject()
+ mockPods := testapps.MockReplicationComponentPods(nil, testCtx, sts, clusterObj.Name, compDefName, nil)
+ Expect(testapps.ChangeObjStatus(&testCtx, sts, func() {
+ testk8s.MockStatefulSetReady(sts)
+ })).ShouldNot(HaveOccurred())
+ Expect(testapps.ChangeObjStatus(&testCtx, rsm, func() {
+ testk8s.MockRSMReady(rsm, mockPods...)
+ })).ShouldNot(HaveOccurred())
Eventually(testapps.GetClusterPhase(&testCtx, clusterKey)).Should(Equal(appsv1alpha1.RunningClusterPhase))
})
})
@@ -2781,13 +2457,13 @@ var _ = Describe("Cluster Controller", func() {
// })).Should(Succeed())
By("test when clusterVersion not Available")
- _ = testapps.CreateConsensusMysqlClusterDef(&testCtx, clusterDefNameRand, consensusCompDefName)
clusterVersion := testapps.CreateConsensusMysqlClusterVersion(&testCtx, clusterDefNameRand, clusterVersionNameRand, consensusCompDefName)
clusterVersionKey := client.ObjectKeyFromObject(clusterVersion)
// mock clusterVersion unavailable
Expect(testapps.GetAndChangeObj(&testCtx, clusterVersionKey, func(clusterVersion *appsv1alpha1.ClusterVersion) {
clusterVersion.Spec.ComponentVersions[0].ComponentDefRef = "test-n"
})()).ShouldNot(HaveOccurred())
+ _ = testapps.CreateConsensusMysqlClusterDef(&testCtx, clusterDefNameRand, consensusCompDefName)
Eventually(testapps.CheckObj(&testCtx, clusterVersionKey, func(g Gomega, clusterVersion *appsv1alpha1.ClusterVersion) {
g.Expect(clusterVersion.Status.Phase).Should(Equal(appsv1alpha1.UnavailablePhase))
@@ -2882,39 +2558,13 @@ var _ = Describe("Cluster Controller", func() {
Name: clusterKey.Name + "-" + consensusCompName,
}
- if intctrlutil.IsRSMEnabled() {
- By("checking workload exists")
- Eventually(testapps.CheckObjExists(&testCtx, workloadKey, &workloads.ReplicatedStateMachine{}, true)).Should(Succeed())
-
- finalizerName := "test/finalizer"
- By("set finalizer for workload to prevent it from deletion")
- Expect(testapps.GetAndChangeObj(&testCtx, workloadKey, func(wl *workloads.ReplicatedStateMachine) {
- wl.ObjectMeta.Finalizers = append(wl.ObjectMeta.Finalizers, finalizerName)
- })()).ShouldNot(HaveOccurred())
-
- By("Delete the cluster")
- testapps.DeleteObject(&testCtx, clusterKey, &appsv1alpha1.Cluster{})
-
- By("checking cluster keep existing")
- Consistently(testapps.CheckObjExists(&testCtx, clusterKey, &appsv1alpha1.Cluster{}, true)).Should(Succeed())
-
- By("remove finalizer of sts to get it deleted")
- Expect(testapps.GetAndChangeObj(&testCtx, workloadKey, func(wl *workloads.ReplicatedStateMachine) {
- wl.ObjectMeta.Finalizers = nil
- })()).ShouldNot(HaveOccurred())
-
- By("Wait for the cluster to terminate")
- Eventually(testapps.CheckObjExists(&testCtx, clusterKey, &appsv1alpha1.Cluster{}, false)).Should(Succeed())
- return
- }
-
- By("checking sts exists")
- Eventually(testapps.CheckObjExists(&testCtx, workloadKey, &appsv1.StatefulSet{}, true)).Should(Succeed())
+ By("checking workload exists")
+ Eventually(testapps.CheckObjExists(&testCtx, workloadKey, &workloads.ReplicatedStateMachine{}, true)).Should(Succeed())
finalizerName := "test/finalizer"
- By("set finalizer for sts to prevent it from deletion")
- Expect(testapps.GetAndChangeObj(&testCtx, workloadKey, func(sts *appsv1.StatefulSet) {
- sts.ObjectMeta.Finalizers = append(sts.ObjectMeta.Finalizers, finalizerName)
+ By("set finalizer for workload to prevent it from deletion")
+ Expect(testapps.GetAndChangeObj(&testCtx, workloadKey, func(wl *workloads.ReplicatedStateMachine) {
+ wl.ObjectMeta.Finalizers = append(wl.ObjectMeta.Finalizers, finalizerName)
})()).ShouldNot(HaveOccurred())
By("Delete the cluster")
@@ -2924,8 +2574,8 @@ var _ = Describe("Cluster Controller", func() {
Consistently(testapps.CheckObjExists(&testCtx, clusterKey, &appsv1alpha1.Cluster{}, true)).Should(Succeed())
By("remove finalizer of sts to get it deleted")
- Expect(testapps.GetAndChangeObj(&testCtx, workloadKey, func(sts *appsv1.StatefulSet) {
- sts.ObjectMeta.Finalizers = nil
+ Expect(testapps.GetAndChangeObj(&testCtx, workloadKey, func(wl *workloads.ReplicatedStateMachine) {
+ wl.ObjectMeta.Finalizers = nil
})()).ShouldNot(HaveOccurred())
By("Wait for the cluster to terminate")
@@ -2940,9 +2590,13 @@ func createBackupPolicyTpl(clusterDefObj *appsv1alpha1.ClusterDefinition) {
AddLabels(constant.ClusterDefLabelKey, clusterDefObj.Name).
SetClusterDefRef(clusterDefObj.Name)
for _, v := range clusterDefObj.Spec.ComponentDefs {
- bpt = bpt.AddBackupPolicy(v.Name).AddSnapshotPolicy().SetSchedule("0 0 * * *", false)
- bpt = bpt.AddDatafilePolicy().SetSchedule("0 0 * * *", false)
- bpt = bpt.AddIncrementalPolicy().SetSchedule("0 0 * * *", false)
+ bpt = bpt.AddBackupPolicy(v.Name).
+ AddBackupMethod(backupMethodName, false, actionSetName).
+ SetBackupMethodVolumeMounts("data", "/data").
+ AddBackupMethod(vsBackupMethodName, true, vsActionSetName).
+ SetBackupMethodVolumes([]string{"data"}).
+ AddSchedule(backupMethodName, "0 0 * * *", true).
+ AddSchedule(vsBackupMethodName, "0 0 * * *", true)
switch v.WorkloadType {
case appsv1alpha1.Consensus:
bpt.SetTargetRole("leader")
@@ -2971,3 +2625,28 @@ func outOfOrderEqualFunc[E1, E2 any](s1 []E1, s2 []E2, eq func(E1, E2) bool) boo
}
return true
}
+
+func mockRestoreCompleted(ml client.MatchingLabels) {
+ restoreList := dpv1alpha1.RestoreList{}
+ Expect(testCtx.Cli.List(testCtx.Ctx, &restoreList, ml)).Should(Succeed())
+ for _, rs := range restoreList.Items {
+ err := testapps.GetAndChangeObjStatus(&testCtx, client.ObjectKeyFromObject(&rs), func(res *dpv1alpha1.Restore) {
+ res.Status.Phase = dpv1alpha1.RestorePhaseCompleted
+ })()
+ Expect(client.IgnoreNotFound(err)).ShouldNot(HaveOccurred())
+ }
+}
+
+func checkRestoreAndSetCompleted(clusterKey types.NamespacedName, compName string, scaleOutReplicas int) {
+ By("Checking restore CR created")
+ ml := client.MatchingLabels{
+ constant.AppInstanceLabelKey: clusterKey.Name,
+ constant.KBAppComponentLabelKey: compName,
+ constant.KBManagedByKey: "cluster",
+ }
+ Eventually(testapps.List(&testCtx, generics.RestoreSignature,
+ ml, client.InNamespace(clusterKey.Namespace))).Should(HaveLen(scaleOutReplicas))
+
+ By("Mocking restore phase to succeeded")
+ mockRestoreCompleted(ml)
+}
diff --git a/controllers/apps/cluster_plan_builder_test.go b/controllers/apps/cluster_plan_builder_test.go
index 468b4cc208a..2e7fc946b3d 100644
--- a/controllers/apps/cluster_plan_builder_test.go
+++ b/controllers/apps/cluster_plan_builder_test.go
@@ -62,7 +62,7 @@ var _ = Describe("cluster plan builder test", func() {
testapps.ClearResourcesWithRemoveFinalizerOption(&testCtx, generics.VolumeSnapshotSignature, true, inNS)
// non-namespaced
testapps.ClearResources(&testCtx, generics.BackupPolicyTemplateSignature, ml)
- testapps.ClearResources(&testCtx, generics.BackupToolSignature, ml)
+ testapps.ClearResources(&testCtx, generics.ActionSetSignature, ml)
testapps.ClearResources(&testCtx, generics.StorageClassSignature, ml)
}
diff --git a/controllers/apps/cluster_status_event_handler_test.go b/controllers/apps/cluster_status_event_handler_test.go
index 4b059cafb00..7ee30195921 100644
--- a/controllers/apps/cluster_status_event_handler_test.go
+++ b/controllers/apps/cluster_status_event_handler_test.go
@@ -25,14 +25,12 @@ import (
. "github.com/onsi/ginkgo/v2"
. "github.com/onsi/gomega"
- appsv1 "k8s.io/api/apps/v1"
corev1 "k8s.io/api/core/v1"
"sigs.k8s.io/controller-runtime/pkg/client"
appsv1alpha1 "github.com/apecloud/kubeblocks/apis/apps/v1alpha1"
workloads "github.com/apecloud/kubeblocks/apis/workloads/v1alpha1"
"github.com/apecloud/kubeblocks/internal/constant"
- "github.com/apecloud/kubeblocks/internal/controllerutil"
intctrlutil "github.com/apecloud/kubeblocks/internal/generics"
testapps "github.com/apecloud/kubeblocks/internal/testutil/apps"
)
@@ -115,7 +113,7 @@ var _ = Describe("test cluster Failed/Abnormal phase", func() {
// AddAppInstanceLabel(clusterName).
// AddAppComponentLabel(componentName).
// AddRoleLabel(podRole).
- // AddAppManangedByLabel().
+ // AddAppManagedByLabel().
// AddContainer(corev1.Container{Name: testapps.DefaultMySQLContainerName, Image: testapps.ApeCloudMySQLImage}).
// Create(&testCtx).GetObject()
// }
@@ -181,20 +179,12 @@ var _ = Describe("test cluster Failed/Abnormal phase", func() {
By("watch warning event from StatefulSet, but mismatch condition ")
// wait for StatefulSet created by cluster controller
workloadName := clusterName + "-" + statefulMySQLCompName
- var kd string
- if controllerutil.IsRSMEnabled() {
- kd = constant.RSMKind
- Eventually(testapps.CheckObj(&testCtx, client.ObjectKey{Name: workloadName, Namespace: testCtx.DefaultNamespace},
- func(g Gomega, fetched *workloads.ReplicatedStateMachine) {
- g.Expect(fetched.Generation).To(BeEquivalentTo(1))
- })).Should(Succeed())
- } else {
- kd = constant.StatefulSetKind
- Eventually(testapps.CheckObj(&testCtx, client.ObjectKey{Name: workloadName, Namespace: testCtx.DefaultNamespace},
- func(g Gomega, fetched *appsv1.StatefulSet) {
- g.Expect(fetched.Generation).To(BeEquivalentTo(1))
- })).Should(Succeed())
- }
+ kd := constant.RSMKind
+ Eventually(testapps.CheckObj(&testCtx, client.ObjectKey{Name: workloadName, Namespace: testCtx.DefaultNamespace},
+ func(g Gomega, fetched *workloads.ReplicatedStateMachine) {
+ g.Expect(fetched.Generation).To(BeEquivalentTo(1))
+ })).Should(Succeed())
+
stsInvolvedObject := corev1.ObjectReference{
Name: workloadName,
Kind: kd,
diff --git a/controllers/apps/components/base.go b/controllers/apps/components/base.go
deleted file mode 100644
index 717c7dedced..00000000000
--- a/controllers/apps/components/base.go
+++ /dev/null
@@ -1,652 +0,0 @@
-/*
-Copyright (C) 2022-2023 ApeCloud Co., Ltd
-
-This file is part of KubeBlocks project
-
-This program is free software: you can redistribute it and/or modify
-it under the terms of the GNU Affero General Public License as published by
-the Free Software Foundation, either version 3 of the License, or
-(at your option) any later version.
-
-This program is distributed in the hope that it will be useful
-but WITHOUT ANY WARRANTY; without even the implied warranty of
-MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
-GNU Affero General Public License for more details.
-
-You should have received a copy of the GNU Affero General Public License
-along with this program. If not, see .
-*/
-
-package components
-
-import (
- "context"
- "fmt"
- "reflect"
- "strconv"
- "strings"
- "time"
-
- "golang.org/x/exp/maps"
- "golang.org/x/exp/slices"
- appsv1 "k8s.io/api/apps/v1"
- corev1 "k8s.io/api/core/v1"
- policyv1 "k8s.io/api/policy/v1"
- apierrors "k8s.io/apimachinery/pkg/api/errors"
- metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
- "k8s.io/apimachinery/pkg/runtime"
- "k8s.io/apimachinery/pkg/runtime/schema"
- "k8s.io/client-go/tools/record"
- "sigs.k8s.io/controller-runtime/pkg/client"
- "sigs.k8s.io/controller-runtime/pkg/client/apiutil"
-
- appsv1alpha1 "github.com/apecloud/kubeblocks/apis/apps/v1alpha1"
- dataprotectionv1alpha1 "github.com/apecloud/kubeblocks/apis/dataprotection/v1alpha1"
- "github.com/apecloud/kubeblocks/internal/constant"
- "github.com/apecloud/kubeblocks/internal/controller/component"
- "github.com/apecloud/kubeblocks/internal/controller/graph"
- ictrltypes "github.com/apecloud/kubeblocks/internal/controller/types"
- intctrlutil "github.com/apecloud/kubeblocks/internal/controllerutil"
- "github.com/apecloud/kubeblocks/internal/generics"
-)
-
-type componentBase struct {
- Client client.Client
- Recorder record.EventRecorder
- Cluster *appsv1alpha1.Cluster
- ClusterVersion *appsv1alpha1.ClusterVersion // building config needs the cluster version
- Component *component.SynthesizedComponent // built synthesized component, replace it with component workload proto
- ComponentSet componentSet
- Dag *graph.DAG
- WorkloadVertex *ictrltypes.LifecycleVertex // DAG vertex of main workload object
-}
-
-func (c *componentBase) GetName() string {
- return c.Component.Name
-}
-
-func (c *componentBase) GetNamespace() string {
- return c.Cluster.Namespace
-}
-
-func (c *componentBase) GetClusterName() string {
- return c.Cluster.Name
-}
-
-func (c *componentBase) GetDefinitionName() string {
- return c.Component.ComponentDef
-}
-
-func (c *componentBase) GetCluster() *appsv1alpha1.Cluster {
- return c.Cluster
-}
-
-func (c *componentBase) GetClusterVersion() *appsv1alpha1.ClusterVersion {
- return c.ClusterVersion
-}
-
-func (c *componentBase) GetSynthesizedComponent() *component.SynthesizedComponent {
- return c.Component
-}
-
-func (c *componentBase) GetConsensusSpec() *appsv1alpha1.ConsensusSetSpec {
- return c.Component.ConsensusSpec
-}
-
-func (c *componentBase) GetMatchingLabels() client.MatchingLabels {
- return client.MatchingLabels{
- constant.AppManagedByLabelKey: constant.AppName,
- constant.AppInstanceLabelKey: c.GetClusterName(),
- constant.KBAppComponentLabelKey: c.GetName(),
- }
-}
-
-func (c *componentBase) GetPhase() appsv1alpha1.ClusterComponentPhase {
- if c.Cluster.Status.Components == nil {
- return ""
- }
- if _, ok := c.Cluster.Status.Components[c.GetName()]; !ok {
- return ""
- }
- return c.Cluster.Status.Components[c.GetName()].Phase
-}
-
-func (c *componentBase) SetWorkload(obj client.Object, action *ictrltypes.LifecycleAction, parent *ictrltypes.LifecycleVertex) {
- c.WorkloadVertex = c.AddResource(obj, action, parent)
-}
-
-func (c *componentBase) AddResource(obj client.Object, action *ictrltypes.LifecycleAction,
- parent *ictrltypes.LifecycleVertex) *ictrltypes.LifecycleVertex {
- if obj == nil {
- panic("try to add nil object")
- }
- vertex := &ictrltypes.LifecycleVertex{
- Obj: obj,
- Action: action,
- }
- c.Dag.AddVertex(vertex)
-
- if parent != nil {
- c.Dag.Connect(parent, vertex)
- }
- return vertex
-}
-
-func (c *componentBase) CreateResource(obj client.Object, parent *ictrltypes.LifecycleVertex) *ictrltypes.LifecycleVertex {
- return ictrltypes.LifecycleObjectCreate(c.Dag, obj, parent)
-}
-
-func (c *componentBase) DeleteResource(obj client.Object, parent *ictrltypes.LifecycleVertex) *ictrltypes.LifecycleVertex {
- return ictrltypes.LifecycleObjectDelete(c.Dag, obj, parent)
-}
-
-func (c *componentBase) UpdateResource(obj client.Object, parent *ictrltypes.LifecycleVertex) *ictrltypes.LifecycleVertex {
- return ictrltypes.LifecycleObjectUpdate(c.Dag, obj, parent)
-}
-
-func (c *componentBase) PatchResource(obj client.Object, objCopy client.Object, parent *ictrltypes.LifecycleVertex) *ictrltypes.LifecycleVertex {
- return ictrltypes.LifecycleObjectPatch(c.Dag, obj, objCopy, parent)
-}
-
-func (c *componentBase) NoopResource(obj client.Object, parent *ictrltypes.LifecycleVertex) *ictrltypes.LifecycleVertex {
- return ictrltypes.LifecycleObjectNoop(c.Dag, obj, parent)
-}
-
-// ValidateObjectsAction validates the action of objects in dag has been determined.
-func (c *componentBase) ValidateObjectsAction() error {
- for _, v := range c.Dag.Vertices() {
- node, ok := v.(*ictrltypes.LifecycleVertex)
- if !ok {
- return fmt.Errorf("unexpected vertex type, cluster: %s, component: %s, vertex: %T",
- c.GetClusterName(), c.GetName(), v)
- }
- if node.Obj == nil {
- return fmt.Errorf("unexpected nil vertex object, cluster: %s, component: %s, vertex: %T",
- c.GetClusterName(), c.GetName(), v)
- }
- if node.Action == nil {
- return fmt.Errorf("unexpected nil vertex action, cluster: %s, component: %s, vertex: %T",
- c.GetClusterName(), c.GetName(), v)
- }
- }
- return nil
-}
-
-// ResolveObjectsAction resolves the action of objects in dag to guarantee that all object actions will be determined.
-func (c *componentBase) ResolveObjectsAction(reqCtx intctrlutil.RequestCtx, cli client.Client) error {
- snapshot, err := readCacheSnapshot(reqCtx, cli, c.GetCluster())
- if err != nil {
- return err
- }
- for _, v := range c.Dag.Vertices() {
- node, ok := v.(*ictrltypes.LifecycleVertex)
- if !ok {
- return fmt.Errorf("unexpected vertex type, cluster: %s, component: %s, vertex: %T",
- c.GetClusterName(), c.GetName(), v)
- }
- if node.Action == nil {
- if action, err := resolveObjectAction(snapshot, node, cli.Scheme()); err != nil {
- return err
- } else {
- node.Action = action
- }
- }
- }
- if c.GetCluster().IsStatusUpdating() {
- for _, vertex := range c.Dag.Vertices() {
- v, _ := vertex.(*ictrltypes.LifecycleVertex)
- // TODO(refactor): fix me, this is a workaround for h-scaling to update stateful set.
- if _, ok := v.Obj.(*appsv1.StatefulSet); !ok {
- v.Immutable = true
- }
- }
- }
- return c.ValidateObjectsAction()
-}
-
-func (c *componentBase) UpdatePDB(reqCtx intctrlutil.RequestCtx, cli client.Client) error {
- pdbObjList, err := listObjWithLabelsInNamespace(reqCtx.Ctx, cli, generics.PodDisruptionBudgetSignature, c.GetNamespace(), c.GetMatchingLabels())
- if err != nil && !apierrors.IsNotFound(err) {
- return err
- }
- for _, v := range ictrltypes.FindAll[*policyv1.PodDisruptionBudget](c.Dag) {
- node := v.(*ictrltypes.LifecycleVertex)
- pdbProto := node.Obj.(*policyv1.PodDisruptionBudget)
-
- if pos := slices.IndexFunc(pdbObjList, func(pdbObj *policyv1.PodDisruptionBudget) bool {
- return pdbObj.GetName() == pdbProto.GetName()
- }); pos < 0 {
- node.Action = ictrltypes.ActionCreatePtr() // TODO: Create or Noop?
- } else {
- pdbObj := pdbObjList[pos]
- if !reflect.DeepEqual(pdbObj.Spec, pdbProto.Spec) {
- pdbObj.Spec = pdbProto.Spec
- node.Obj = pdbObj
- node.Action = ictrltypes.ActionUpdatePtr()
- }
- }
- }
- return nil
-}
-
-func (c *componentBase) UpdateService(reqCtx intctrlutil.RequestCtx, cli client.Client) error {
- svcObjList, err := listObjWithLabelsInNamespace(reqCtx.Ctx, cli, generics.ServiceSignature, c.GetNamespace(), c.GetMatchingLabels())
- if err != nil {
- return client.IgnoreNotFound(err)
- }
-
- svcProtoList := ictrltypes.FindAll[*corev1.Service](c.Dag)
-
- // create new services or update existing services
- for _, vertex := range svcProtoList {
- node, _ := vertex.(*ictrltypes.LifecycleVertex)
- svcProto, _ := node.Obj.(*corev1.Service)
-
- if pos := slices.IndexFunc(svcObjList, func(svc *corev1.Service) bool {
- return svc.GetName() == svcProto.GetName()
- }); pos < 0 {
- node.Action = ictrltypes.ActionCreatePtr()
- } else {
- svcObj := svcObjList[pos]
- // remove original monitor annotations
- if len(svcObj.Annotations) > 0 {
- maps.DeleteFunc(svcObj.Annotations, func(k, v string) bool {
- return strings.HasPrefix(k, "monitor.kubeblocks.io")
- })
- }
- mergeAnnotations(svcObj.Annotations, &svcProto.Annotations)
- svcObj.Annotations = svcProto.Annotations
- svcObj.Spec = svcProto.Spec
- node.Obj = svcObj
- node.Action = ictrltypes.ActionUpdatePtr()
- }
- }
-
- // delete useless services
- for _, svc := range svcObjList {
- if pos := slices.IndexFunc(svcProtoList, func(vertex graph.Vertex) bool {
- node, _ := vertex.(*ictrltypes.LifecycleVertex)
- svcProto, _ := node.Obj.(*corev1.Service)
- return svcProto.GetName() == svc.GetName()
- }); pos < 0 {
- c.DeleteResource(svc, nil)
- }
- }
- return nil
-}
-
-// SetStatusPhase sets the cluster component phase and messages conditionally.
-func (c *componentBase) SetStatusPhase(phase appsv1alpha1.ClusterComponentPhase,
- statusMessage appsv1alpha1.ComponentMessageMap, phaseTransitionMsg string) {
- updatefn := func(status *appsv1alpha1.ClusterComponentStatus) error {
- if status.Phase == phase {
- return nil
- }
- status.Phase = phase
- if status.Message == nil {
- status.Message = statusMessage
- } else {
- for k, v := range statusMessage {
- status.Message[k] = v
- }
- }
- return nil
- }
- if err := c.updateStatus(phaseTransitionMsg, updatefn); err != nil {
- panic(fmt.Sprintf("unexpected error occurred while updating component status: %s", err.Error()))
- }
-}
-
-func (c *componentBase) StatusWorkload(reqCtx intctrlutil.RequestCtx, cli client.Client, obj client.Object, txn *statusReconciliationTxn) error {
- // if reflect.ValueOf(obj).Kind() == reflect.Ptr && reflect.ValueOf(obj).IsNil() {
- // return nil
- // }
-
- pods, err := listPodOwnedByComponent(reqCtx.Ctx, cli, c.GetNamespace(), c.GetMatchingLabels())
- if err != nil {
- return err
- }
-
- isRunning, err := c.ComponentSet.IsRunning(reqCtx.Ctx, obj)
- if err != nil {
- return err
- }
-
- var podsReady *bool
- if c.Component.Replicas > 0 {
- podsReadyForComponent, err := c.ComponentSet.PodsReady(reqCtx.Ctx, obj)
- if err != nil {
- return err
- }
- podsReady = &podsReadyForComponent
- }
-
- hasFailedPodTimedOut := false
- timedOutPodStatusMessage := appsv1alpha1.ComponentMessageMap{}
- var delayedRequeueError error
- isLatestWorkload := obj.GetAnnotations()[constant.KubeBlocksGenerationKey] == strconv.FormatInt(c.Cluster.Generation, 10)
- // check if it is the latest obj after cluster does updates.
- if !isRunning && !appsv1alpha1.ComponentPodsAreReady(podsReady) && isLatestWorkload {
- var requeueAfter time.Duration
- if hasFailedPodTimedOut, timedOutPodStatusMessage, requeueAfter = hasFailedAndTimedOutPod(pods); requeueAfter != 0 {
- delayedRequeueError = intctrlutil.NewDelayedRequeueError(requeueAfter, "requeue for workload status to reconcile.")
- }
- }
-
- phase, statusMessage, err := c.buildStatus(reqCtx.Ctx, pods, isRunning, podsReady, hasFailedPodTimedOut, timedOutPodStatusMessage)
- if err != nil {
- if !intctrlutil.IsDelayedRequeueError(err) {
- return err
- }
- delayedRequeueError = err
- }
-
- phaseTransitionCondMsg := ""
- if podsReady == nil {
- phaseTransitionCondMsg = fmt.Sprintf("Running: %v, PodsReady: nil, PodsTimedout: %v", isRunning, hasFailedPodTimedOut)
- } else {
- phaseTransitionCondMsg = fmt.Sprintf("Running: %v, PodsReady: %v, PodsTimedout: %v", isRunning, *podsReady, hasFailedPodTimedOut)
- }
-
- updatefn := func(status *appsv1alpha1.ClusterComponentStatus) error {
- if phase != "" {
- status.Phase = phase
- }
- status.SetMessage(statusMessage)
- if !appsv1alpha1.ComponentPodsAreReady(podsReady) {
- status.PodsReadyTime = nil
- } else if !appsv1alpha1.ComponentPodsAreReady(status.PodsReady) {
- // set podsReadyTime when pods of component are ready at the moment.
- status.PodsReadyTime = &metav1.Time{Time: time.Now()}
- }
- status.PodsReady = podsReady
- return nil
- }
-
- if txn != nil {
- txn.propose(phase, func() {
- if err = c.updateStatus(phaseTransitionCondMsg, updatefn); err != nil {
- panic(fmt.Sprintf("unexpected error occurred while updating component status: %s", err.Error()))
- }
- })
- return delayedRequeueError
- }
- // TODO(refactor): wait = true to requeue.
- if err = c.updateStatus(phaseTransitionCondMsg, updatefn); err != nil {
- return err
- }
- return delayedRequeueError
-}
-
-func (c *componentBase) buildStatus(ctx context.Context, pods []*corev1.Pod, isRunning bool, podsReady *bool,
- hasFailedPodTimedOut bool, timedOutPodStatusMessage appsv1alpha1.ComponentMessageMap) (appsv1alpha1.ClusterComponentPhase, appsv1alpha1.ComponentMessageMap, error) {
- var (
- err error
- phase appsv1alpha1.ClusterComponentPhase
- statusMessage appsv1alpha1.ComponentMessageMap
- )
- if isRunning {
- if c.Component.Replicas == 0 {
- // if replicas number of component is zero, the component has stopped.
- // 'Stopped' is a special 'Running' status for workload(StatefulSet/Deployment).
- phase = appsv1alpha1.StoppedClusterCompPhase
- } else {
- // change component phase to Running when workloads of component are running.
- phase = appsv1alpha1.RunningClusterCompPhase
- }
- return phase, statusMessage, nil
- }
-
- if appsv1alpha1.ComponentPodsAreReady(podsReady) {
- // check if the role probe timed out when component phase is not Running but all pods of component are ready.
- phase, statusMessage = c.ComponentSet.GetPhaseWhenPodsReadyAndProbeTimeout(pods)
- // if component is not running and probe is not timed out, requeue.
- if phase == "" {
- c.Recorder.Event(c.Cluster, corev1.EventTypeNormal, "WaitingForProbeSuccess", "Waiting for probe success")
- return phase, statusMessage, intctrlutil.NewDelayedRequeueError(time.Second*10, "Waiting for probe success")
- }
- return phase, statusMessage, nil
- }
-
- // get the phase if failed pods have timed out or the pods are not running when there are no changes to the component.
- originPhaseIsUpRunning := slices.Contains(appsv1alpha1.GetComponentUpRunningPhase(), c.GetPhase())
- if hasFailedPodTimedOut || originPhaseIsUpRunning {
- phase, statusMessage, err = c.ComponentSet.GetPhaseWhenPodsNotReady(ctx, c.GetName(), originPhaseIsUpRunning)
- if err != nil {
- return "", nil, err
- }
- }
- if statusMessage == nil {
- statusMessage = timedOutPodStatusMessage
- } else {
- for k, v := range timedOutPodStatusMessage {
- statusMessage[k] = v
- }
- }
- return phase, statusMessage, nil
-}
-
-// updateStatus updates the cluster component status by @updatefn, with additional message to explain the transition occurred.
-func (c *componentBase) updateStatus(phaseTransitionMsg string, updatefn func(status *appsv1alpha1.ClusterComponentStatus) error) error {
- if updatefn == nil {
- return nil
- }
-
- status := c.getComponentStatus()
- phase := status.Phase
- err := updatefn(&status)
- if err != nil {
- return err
- }
- c.Cluster.Status.Components[c.GetName()] = status
-
- if phase != status.Phase {
- // TODO: logging the event
- if c.Recorder != nil && phaseTransitionMsg != "" {
- c.Recorder.Eventf(c.Cluster, corev1.EventTypeNormal, ComponentPhaseTransition, phaseTransitionMsg)
- }
- }
-
- return nil
-}
-
-func (c *componentBase) getComponentStatus() appsv1alpha1.ClusterComponentStatus {
- if c.Cluster.Status.Components == nil {
- c.Cluster.Status.Components = make(map[string]appsv1alpha1.ClusterComponentStatus)
- }
- if _, ok := c.Cluster.Status.Components[c.GetName()]; !ok {
- c.Cluster.Status.Components[c.GetName()] = appsv1alpha1.ClusterComponentStatus{}
- }
- return c.Cluster.Status.Components[c.GetName()]
-}
-
-// hasFailedAndTimedOutPod returns whether the pods of components are still failed after a PodFailedTimeout period.
-func hasFailedAndTimedOutPod(pods []*corev1.Pod) (bool, appsv1alpha1.ComponentMessageMap, time.Duration) {
- var (
- hasTimedOutPod bool
- messages = appsv1alpha1.ComponentMessageMap{}
- hasFailedPod bool
- requeueAfter time.Duration
- )
- for _, pod := range pods {
- isFailed, isTimedOut, messageStr := IsPodFailedAndTimedOut(pod)
- if !isFailed {
- continue
- }
- if isTimedOut {
- hasTimedOutPod = true
- messages.SetObjectMessage(pod.Kind, pod.Name, messageStr)
- } else {
- hasFailedPod = true
- }
- }
- if hasFailedPod && !hasTimedOutPod {
- requeueAfter = PodContainerFailedTimeout
- }
- return hasTimedOutPod, messages, requeueAfter
-}
-
-// isPodScheduledFailedAndTimedOut checks whether the unscheduled pod has timed out.
-func isPodScheduledFailedAndTimedOut(pod *corev1.Pod) (bool, bool, string) {
- for _, cond := range pod.Status.Conditions {
- if cond.Type != corev1.PodScheduled {
- continue
- }
- if cond.Status == corev1.ConditionTrue {
- return false, false, ""
- }
- return true, time.Now().After(cond.LastTransitionTime.Add(PodScheduledFailedTimeout)), cond.Message
- }
- return false, false, ""
-}
-
-// IsPodFailedAndTimedOut checks if the pod is failed and timed out.
-func IsPodFailedAndTimedOut(pod *corev1.Pod) (bool, bool, string) {
- if isFailed, isTimedOut, message := isPodScheduledFailedAndTimedOut(pod); isFailed {
- return isFailed, isTimedOut, message
- }
- initContainerFailed, message := isAnyContainerFailed(pod.Status.InitContainerStatuses)
- if initContainerFailed {
- return initContainerFailed, isContainerFailedAndTimedOut(pod, corev1.PodInitialized), message
- }
- containerFailed, message := isAnyContainerFailed(pod.Status.ContainerStatuses)
- if containerFailed {
- return containerFailed, isContainerFailedAndTimedOut(pod, corev1.ContainersReady), message
- }
- return false, false, ""
-}
-
-// isAnyContainerFailed checks whether any container in the list is failed.
-func isAnyContainerFailed(containersStatus []corev1.ContainerStatus) (bool, string) {
- for _, v := range containersStatus {
- waitingState := v.State.Waiting
- if waitingState != nil && waitingState.Message != "" {
- return true, waitingState.Message
- }
- terminatedState := v.State.Terminated
- if terminatedState != nil && terminatedState.Message != "" {
- return true, terminatedState.Message
- }
- }
- return false, ""
-}
-
-// isContainerFailedAndTimedOut checks whether the failed container has timed out.
-func isContainerFailedAndTimedOut(pod *corev1.Pod, podConditionType corev1.PodConditionType) bool {
- containerReadyCondition := intctrlutil.GetPodCondition(&pod.Status, podConditionType)
- if containerReadyCondition == nil || containerReadyCondition.LastTransitionTime.IsZero() {
- return false
- }
- return time.Now().After(containerReadyCondition.LastTransitionTime.Add(PodContainerFailedTimeout))
-}
-
-type gvkName struct {
- gvk schema.GroupVersionKind
- ns, name string
-}
-
-type clusterSnapshot map[gvkName]client.Object
-
-func getGVKName(object client.Object, scheme *runtime.Scheme) (*gvkName, error) {
- gvk, err := apiutil.GVKForObject(object, scheme)
- if err != nil {
- return nil, err
- }
- return &gvkName{
- gvk: gvk,
- ns: object.GetNamespace(),
- name: object.GetName(),
- }, nil
-}
-
-func isOwnerOf(owner, obj client.Object, scheme *runtime.Scheme) bool {
- ro, ok := owner.(runtime.Object)
- if !ok {
- return false
- }
- gvk, err := apiutil.GVKForObject(ro, scheme)
- if err != nil {
- return false
- }
- ref := metav1.OwnerReference{
- APIVersion: gvk.GroupVersion().String(),
- Kind: gvk.Kind,
- UID: owner.GetUID(),
- Name: owner.GetName(),
- }
- owners := obj.GetOwnerReferences()
- referSameObject := func(a, b metav1.OwnerReference) bool {
- aGV, err := schema.ParseGroupVersion(a.APIVersion)
- if err != nil {
- return false
- }
-
- bGV, err := schema.ParseGroupVersion(b.APIVersion)
- if err != nil {
- return false
- }
-
- return aGV.Group == bGV.Group && a.Kind == b.Kind && a.Name == b.Name
- }
- for _, ownerRef := range owners {
- if referSameObject(ownerRef, ref) {
- return true
- }
- }
- return false
-}
-
-func ownedKinds() []client.ObjectList {
- return []client.ObjectList{
- &appsv1.StatefulSetList{},
- &appsv1.DeploymentList{},
- &corev1.ServiceList{},
- &corev1.SecretList{},
- &corev1.ConfigMapList{},
- &corev1.PersistentVolumeClaimList{}, // TODO(merge): remove it?
- &policyv1.PodDisruptionBudgetList{},
- &dataprotectionv1alpha1.BackupPolicyList{},
- }
-}
-
-// read all objects owned by component
-func readCacheSnapshot(reqCtx intctrlutil.RequestCtx, cli client.Client, cluster *appsv1alpha1.Cluster) (clusterSnapshot, error) {
- // list what kinds of object cluster owns
- kinds := ownedKinds()
- snapshot := make(clusterSnapshot)
- ml := client.MatchingLabels{constant.AppInstanceLabelKey: cluster.GetName()}
- inNS := client.InNamespace(cluster.Namespace)
- for _, list := range kinds {
- if err := cli.List(reqCtx.Ctx, list, inNS, ml); err != nil {
- return nil, err
- }
- // reflect get list.Items
- items := reflect.ValueOf(list).Elem().FieldByName("Items")
- l := items.Len()
- for i := 0; i < l; i++ {
- // get the underlying object
- object := items.Index(i).Addr().Interface().(client.Object)
- // put to snapshot if owned by our cluster
- if isOwnerOf(cluster, object, cli.Scheme()) {
- name, err := getGVKName(object, cli.Scheme())
- if err != nil {
- return nil, err
- }
- snapshot[*name] = object
- }
- }
- }
- return snapshot, nil
-}
-
-func resolveObjectAction(snapshot clusterSnapshot, vertex *ictrltypes.LifecycleVertex, scheme *runtime.Scheme) (*ictrltypes.LifecycleAction, error) {
- gvk, err := getGVKName(vertex.Obj, scheme)
- if err != nil {
- return nil, err
- }
- if _, ok := snapshot[*gvk]; ok {
- return ictrltypes.ActionNoopPtr(), nil
- } else {
- return ictrltypes.ActionCreatePtr(), nil
- }
-}
diff --git a/controllers/apps/components/base_stateful.go b/controllers/apps/components/base_stateful.go
deleted file mode 100644
index 6e4678ef32e..00000000000
--- a/controllers/apps/components/base_stateful.go
+++ /dev/null
@@ -1,993 +0,0 @@
-/*
-Copyright (C) 2022-2023 ApeCloud Co., Ltd
-
-This file is part of KubeBlocks project
-
-This program is free software: you can redistribute it and/or modify
-it under the terms of the GNU Affero General Public License as published by
-the Free Software Foundation, either version 3 of the License, or
-(at your option) any later version.
-
-This program is distributed in the hope that it will be useful
-but WITHOUT ANY WARRANTY; without even the implied warranty of
-MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
-GNU Affero General Public License for more details.
-
-You should have received a copy of the GNU Affero General Public License
-along with this program. If not, see .
-*/
-
-package components
-
-import (
- "fmt"
- "reflect"
- "strconv"
- "strings"
- "time"
-
- "golang.org/x/exp/maps"
- appsv1 "k8s.io/api/apps/v1"
- corev1 "k8s.io/api/core/v1"
- apierrors "k8s.io/apimachinery/pkg/api/errors"
- metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
- "k8s.io/apimachinery/pkg/types"
- "k8s.io/apimachinery/pkg/util/intstr"
- "k8s.io/apimachinery/pkg/util/sets"
- "k8s.io/kubectl/pkg/util/podutils"
- "sigs.k8s.io/controller-runtime/pkg/client"
-
- appsv1alpha1 "github.com/apecloud/kubeblocks/apis/apps/v1alpha1"
- workloads "github.com/apecloud/kubeblocks/apis/workloads/v1alpha1"
- cfgcore "github.com/apecloud/kubeblocks/internal/configuration/core"
- "github.com/apecloud/kubeblocks/internal/configuration/util"
- "github.com/apecloud/kubeblocks/internal/constant"
- "github.com/apecloud/kubeblocks/internal/controller/graph"
- ictrltypes "github.com/apecloud/kubeblocks/internal/controller/types"
- intctrlutil "github.com/apecloud/kubeblocks/internal/controllerutil"
- "github.com/apecloud/kubeblocks/internal/generics"
- viper "github.com/apecloud/kubeblocks/internal/viperx"
- lorry "github.com/apecloud/kubeblocks/lorry/client"
-)
-
-// rsmComponentBase as a base class for single rsm based component (stateful & replication & consensus).
-type rsmComponentBase struct {
- componentBase
- // runningWorkload can be nil, and the replicas of workload can be nil (zero)
- runningWorkload *workloads.ReplicatedStateMachine
-}
-
-func (c *rsmComponentBase) init(reqCtx intctrlutil.RequestCtx, cli client.Client, builder componentWorkloadBuilder, load bool) error {
- var err error
- if builder != nil {
- if err = builder.BuildEnv().
- BuildWorkload().
- BuildPDB().
- BuildConfig().
- BuildTLSVolume().
- BuildVolumeMount().
- BuildTLSCert().
- Complete(); err != nil {
- return err
- }
- }
- if load {
- c.runningWorkload, err = c.loadRunningWorkload(reqCtx, cli)
- if err != nil {
- return err
- }
- }
- return nil
-}
-
-func (c *rsmComponentBase) loadRunningWorkload(reqCtx intctrlutil.RequestCtx, cli client.Client) (*workloads.ReplicatedStateMachine, error) {
- rsmList, err := listRSMOwnedByComponent(reqCtx.Ctx, cli, c.GetNamespace(), c.GetMatchingLabels())
- if err != nil {
- return nil, err
- }
- cnt := len(rsmList)
- switch {
- case cnt == 0:
- return nil, nil
- case cnt == 1:
- return rsmList[0], nil
- default:
- return nil, fmt.Errorf("more than one workloads found for the component, cluster: %s, component: %s, cnt: %d",
- c.GetClusterName(), c.GetName(), cnt)
- }
-}
-
-func (c *rsmComponentBase) GetBuiltObjects(builder componentWorkloadBuilder) ([]client.Object, error) {
- dagSnapshot := c.Dag
- defer func() {
- c.Dag = dagSnapshot
- }()
-
- c.Dag = graph.NewDAG()
- if err := c.init(intctrlutil.RequestCtx{}, nil, builder, false); err != nil {
- return nil, err
- }
-
- objs := make([]client.Object, 0)
- for _, v := range c.Dag.Vertices() {
- if vv, ok := v.(*ictrltypes.LifecycleVertex); ok {
- objs = append(objs, vv.Obj)
- }
- }
- return objs, nil
-}
-
-func (c *rsmComponentBase) Create(reqCtx intctrlutil.RequestCtx, cli client.Client, builder componentWorkloadBuilder) error {
- if err := c.init(reqCtx, cli, builder, false); err != nil {
- return err
- }
-
- if err := c.ValidateObjectsAction(); err != nil {
- return err
- }
-
- return nil
-}
-
-func (c *rsmComponentBase) Delete(reqCtx intctrlutil.RequestCtx, cli client.Client) error {
- // TODO(impl): delete component owned resources
- return nil
-}
-
-func (c *rsmComponentBase) Update(reqCtx intctrlutil.RequestCtx, cli client.Client, builder componentWorkloadBuilder) error {
- if err := c.init(reqCtx, cli, builder, true); err != nil {
- return err
- }
-
- if c.runningWorkload != nil {
- if err := c.Restart(reqCtx, cli); err != nil {
- return err
- }
-
- // cluster.spec.componentSpecs[*].volumeClaimTemplates[*].spec.resources.requests[corev1.ResourceStorage]
- if err := c.ExpandVolume(reqCtx, cli); err != nil {
- return err
- }
-
- // cluster.spec.componentSpecs[*].replicas
- if err := c.HorizontalScale(reqCtx, cli); err != nil {
- return err
- }
- }
-
- if err := c.updateUnderlyingResources(reqCtx, cli, c.runningWorkload); err != nil {
- return err
- }
-
- return c.ResolveObjectsAction(reqCtx, cli)
-}
-
-func (c *rsmComponentBase) Status(reqCtx intctrlutil.RequestCtx, cli client.Client, builder componentWorkloadBuilder) error {
- if err := c.init(reqCtx, cli, builder, true); err != nil {
- return err
- }
- if c.runningWorkload == nil {
- return nil
- }
-
- isDeleting := func() bool {
- return !c.runningWorkload.DeletionTimestamp.IsZero()
- }()
- isZeroReplica := func() bool {
- return (c.runningWorkload.Spec.Replicas == nil || *c.runningWorkload.Spec.Replicas == 0) && c.Component.Replicas == 0
- }()
- pods, err := listPodOwnedByComponent(reqCtx.Ctx, cli, c.GetNamespace(), c.GetMatchingLabels())
- if err != nil {
- return err
- }
- hasComponentPod := func() bool {
- return len(pods) > 0
- }()
- isRunning, err := c.ComponentSet.IsRunning(reqCtx.Ctx, c.runningWorkload)
- if err != nil {
- return err
- }
- isAllConfigSynced := c.isAllConfigSynced(reqCtx, cli)
- hasFailedPod, messages, err := c.hasFailedPod(reqCtx, cli, pods)
- if err != nil {
- return err
- }
- isScaleOutFailed, err := c.isScaleOutFailed(reqCtx, cli)
- if err != nil {
- return err
- }
- hasRunningVolumeExpansion, hasFailedVolumeExpansion, err := c.hasVolumeExpansionRunning(reqCtx, cli)
- if err != nil {
- return err
- }
- hasFailure := func() bool {
- return hasFailedPod || isScaleOutFailed || hasFailedVolumeExpansion
- }()
- isComponentAvailable, err := c.isAvailable(reqCtx, cli, pods)
- if err != nil {
- return err
- }
- isInCreatingPhase := func() bool {
- phase := c.getComponentStatus().Phase
- return phase == "" || phase == appsv1alpha1.CreatingClusterCompPhase
- }()
-
- updatePodsReady := func(ready bool) {
- _ = c.updateStatus("", func(status *appsv1alpha1.ClusterComponentStatus) error {
- // if ready flag not changed, don't update the ready time
- if status.PodsReady != nil && *status.PodsReady == ready {
- return nil
- }
- status.PodsReady = &ready
- if ready {
- time := metav1.Now()
- status.PodsReadyTime = &time
- }
- return nil
- })
- }
-
- podsReady := false
- switch {
- case isDeleting:
- c.SetStatusPhase(appsv1alpha1.DeletingClusterCompPhase, nil, "Component is Deleting")
- case isZeroReplica && hasComponentPod:
- c.SetStatusPhase(appsv1alpha1.StoppingClusterCompPhase, nil, "Component is Stopping")
- podsReady = true
- case isZeroReplica:
- c.SetStatusPhase(appsv1alpha1.StoppedClusterCompPhase, nil, "Component is Stopped")
- podsReady = true
- case isRunning && isAllConfigSynced && !hasRunningVolumeExpansion:
- c.SetStatusPhase(appsv1alpha1.RunningClusterCompPhase, nil, "Component is Running")
- podsReady = true
- case !hasFailure && isInCreatingPhase:
- c.SetStatusPhase(appsv1alpha1.CreatingClusterCompPhase, nil, "Create a new component")
- case !hasFailure:
- c.SetStatusPhase(appsv1alpha1.UpdatingClusterCompPhase, nil, "Component is Updating")
- case !isComponentAvailable:
- c.SetStatusPhase(appsv1alpha1.FailedClusterCompPhase, messages, "Component is Failed")
- default:
- c.SetStatusPhase(appsv1alpha1.AbnormalClusterCompPhase, nil, "unknown")
- }
- updatePodsReady(podsReady)
-
- // works should continue to be done after spec updated.
- if err := c.horizontalScale(reqCtx, cli); err != nil {
- return err
- }
-
- if vertexes, err := c.ComponentSet.HandleRoleChange(reqCtx.Ctx, c.runningWorkload); err != nil {
- return err
- } else {
- for _, v := range vertexes {
- c.Dag.AddVertex(v)
- }
- }
-
- c.updateWorkload(c.runningWorkload)
-
- // update component info to pods' annotations
- if err := updateComponentInfoToPods(reqCtx.Ctx, cli, c.Cluster, c.Component, c.Dag); err != nil {
- return err
- }
-
- // patch the current componentSpec workload's custom labels
- if err := updateCustomLabelToPods(reqCtx.Ctx, cli, c.Cluster, c.Component, c.Dag); err != nil {
- reqCtx.Event(c.Cluster, corev1.EventTypeWarning, "Component Workload Controller PatchWorkloadCustomLabelFailed", err.Error())
- return err
- }
-
- return nil
-}
-
-// isAvailable tells whether the component is basically available, ether working well or in a fragile state:
-// 1. at least one pod is available
-// 2. with latest revision
-// 3. and with leader role label set
-func (c *rsmComponentBase) isAvailable(reqCtx intctrlutil.RequestCtx, cli client.Client, pods []*corev1.Pod) (bool, error) {
- if isLatestRevision, err := IsComponentPodsWithLatestRevision(reqCtx.Ctx, cli, c.Cluster, c.runningWorkload); err != nil {
- return false, err
- } else if !isLatestRevision {
- return false, nil
- }
-
- shouldCheckLeader := func() bool {
- return c.Component.WorkloadType == appsv1alpha1.Consensus || c.Component.WorkloadType == appsv1alpha1.Replication
- }()
- hasLeaderRoleLabel := func(pod *corev1.Pod) bool {
- roleName, ok := pod.Labels[constant.RoleLabelKey]
- if !ok {
- return false
- }
- for _, replicaRole := range c.runningWorkload.Spec.Roles {
- if roleName == replicaRole.Name && replicaRole.IsLeader {
- return true
- }
- }
- return false
- }
- for _, pod := range pods {
- if !podutils.IsPodAvailable(pod, 0, metav1.Time{Time: time.Now()}) {
- continue
- }
- if !shouldCheckLeader {
- continue
- }
- if _, ok := pod.Labels[constant.RoleLabelKey]; ok {
- continue
- }
- if hasLeaderRoleLabel(pod) {
- return true, nil
- }
- }
- return false, nil
-}
-
-func (c *rsmComponentBase) hasFailedPod(reqCtx intctrlutil.RequestCtx, cli client.Client, pods []*corev1.Pod) (bool, appsv1alpha1.ComponentMessageMap, error) {
- if isLatestRevision, err := IsComponentPodsWithLatestRevision(reqCtx.Ctx, cli, c.Cluster, c.runningWorkload); err != nil {
- return false, nil, err
- } else if !isLatestRevision {
- return false, nil, nil
- }
-
- var messages appsv1alpha1.ComponentMessageMap
- // check pod readiness
- hasFailedPod, msg, _ := hasFailedAndTimedOutPod(pods)
- if hasFailedPod {
- messages = msg
- return true, messages, nil
- }
- // check role probe
- if c.Component.WorkloadType != appsv1alpha1.Consensus && c.Component.WorkloadType != appsv1alpha1.Replication {
- return false, messages, nil
- }
- hasProbeTimeout := false
- for _, pod := range pods {
- if _, ok := pod.Labels[constant.RoleLabelKey]; ok {
- continue
- }
- for _, condition := range pod.Status.Conditions {
- if condition.Type != corev1.PodReady || condition.Status != corev1.ConditionTrue {
- continue
- }
- podsReadyTime := &condition.LastTransitionTime
- if isProbeTimeout(c.Component.Probes, podsReadyTime) {
- hasProbeTimeout = true
- if messages == nil {
- messages = appsv1alpha1.ComponentMessageMap{}
- }
- messages.SetObjectMessage(pod.Kind, pod.Name, "Role probe timeout, check whether the application is available")
- }
- }
- }
- return hasProbeTimeout, messages, nil
-}
-
-func (c *rsmComponentBase) isScaleOutFailed(reqCtx intctrlutil.RequestCtx, cli client.Client) (bool, error) {
- if c.runningWorkload.Spec.Replicas == nil {
- return false, nil
- }
- if c.Component.Replicas <= *c.runningWorkload.Spec.Replicas {
- return false, nil
- }
- if c.WorkloadVertex == nil {
- return false, nil
- }
- stsObj := ConvertRSMToSTS(c.runningWorkload)
- rsmProto := c.WorkloadVertex.Obj.(*workloads.ReplicatedStateMachine)
- stsProto := ConvertRSMToSTS(rsmProto)
- backupKey := types.NamespacedName{
- Namespace: stsObj.Namespace,
- Name: stsObj.Name + "-scaling",
- }
- d, err := newDataClone(reqCtx, cli, c.Cluster, c.Component, stsObj, stsProto, backupKey)
- if err != nil {
- return false, err
- }
- if status, err := d.checkBackupStatus(); err != nil {
- return false, err
- } else if status == backupStatusFailed {
- return true, nil
- }
- for _, name := range d.pvcKeysToRestore() {
- if status, err := d.checkRestoreStatus(name); err != nil {
- return false, err
- } else if status == backupStatusFailed {
- return true, nil
- }
- }
- return false, nil
-}
-
-func (c *rsmComponentBase) Restart(reqCtx intctrlutil.RequestCtx, cli client.Client) error {
- return restartPod(&c.runningWorkload.Spec.Template)
-}
-
-func (c *rsmComponentBase) ExpandVolume(reqCtx intctrlutil.RequestCtx, cli client.Client) error {
- for _, vct := range c.runningWorkload.Spec.VolumeClaimTemplates {
- var proto *corev1.PersistentVolumeClaimTemplate
- for _, v := range c.Component.VolumeClaimTemplates {
- if v.Name == vct.Name {
- proto = &v
- break
- }
- }
- // REVIEW: seems we can remove a volume claim from templates at runtime, without any changes and warning messages?
- if proto == nil {
- continue
- }
-
- if err := c.expandVolumes(reqCtx, cli, vct.Name, proto); err != nil {
- return err
- }
- }
- return nil
-}
-
-func (c *rsmComponentBase) expandVolumes(reqCtx intctrlutil.RequestCtx, cli client.Client,
- vctName string, proto *corev1.PersistentVolumeClaimTemplate) error {
- pvcNotFound := false
- for i := *c.runningWorkload.Spec.Replicas - 1; i >= 0; i-- {
- pvc := &corev1.PersistentVolumeClaim{}
- pvcKey := types.NamespacedName{
- Namespace: c.GetNamespace(),
- Name: fmt.Sprintf("%s-%s-%d", vctName, c.runningWorkload.Name, i),
- }
- if err := cli.Get(reqCtx.Ctx, pvcKey, pvc); err != nil {
- if apierrors.IsNotFound(err) {
- pvcNotFound = true
- } else {
- return err
- }
- }
- if err := c.updatePVCSize(reqCtx, cli, pvcKey, pvc, pvcNotFound, proto); err != nil {
- return err
- }
- }
- return nil
-}
-
-func (c *rsmComponentBase) updatePVCSize(reqCtx intctrlutil.RequestCtx, cli client.Client, pvcKey types.NamespacedName,
- pvc *corev1.PersistentVolumeClaim, pvcNotFound bool, vctProto *corev1.PersistentVolumeClaimTemplate) error {
- // reference: https://kubernetes.io/docs/concepts/storage/persistent-volumes/#recovering-from-failure-when-expanding-volumes
- // 1. Mark the PersistentVolume(PV) that is bound to the PersistentVolumeClaim(PVC) with Retain reclaim policy.
- // 2. Delete the PVC. Since PV has Retain reclaim policy - we will not lose any data when we recreate the PVC.
- // 3. Delete the claimRef entry from PV specs, so as new PVC can bind to it. This should make the PV Available.
- // 4. Re-create the PVC with smaller size than PV and set volumeName field of the PVC to the name of the PV. This should bind new PVC to existing PV.
- // 5. Don't forget to restore the reclaim policy of the PV.
- newPVC := pvc.DeepCopy()
- if pvcNotFound {
- newPVC.Name = pvcKey.Name
- newPVC.Namespace = pvcKey.Namespace
- newPVC.SetLabels(vctProto.Labels)
- newPVC.Spec = vctProto.Spec
- ml := client.MatchingLabels{
- constant.PVCNameLabelKey: pvcKey.Name,
- }
- pvList := corev1.PersistentVolumeList{}
- if err := cli.List(reqCtx.Ctx, &pvList, ml); err != nil {
- return err
- }
- for _, pv := range pvList.Items {
- // find pv referenced this pvc
- if pv.Spec.ClaimRef == nil {
- continue
- }
- if pv.Spec.ClaimRef.Name == pvcKey.Name {
- newPVC.Spec.VolumeName = pv.Name
- break
- }
- }
- } else {
- newPVC.Spec.Resources.Requests[corev1.ResourceStorage] = vctProto.Spec.Resources.Requests[corev1.ResourceStorage]
- // delete annotation to make it re-bind
- delete(newPVC.Annotations, "pv.kubernetes.io/bind-completed")
- }
-
- pvNotFound := false
-
- // step 1: update pv to retain
- pv := &corev1.PersistentVolume{}
- pvKey := types.NamespacedName{
- Namespace: pvcKey.Namespace,
- Name: newPVC.Spec.VolumeName,
- }
- if err := cli.Get(reqCtx.Ctx, pvKey, pv); err != nil {
- if apierrors.IsNotFound(err) {
- pvNotFound = true
- } else {
- return err
- }
- }
-
- type pvcRecreateStep int
- const (
- pvPolicyRetainStep pvcRecreateStep = iota
- deletePVCStep
- removePVClaimRefStep
- createPVCStep
- pvRestorePolicyStep
- )
-
- addStepMap := map[pvcRecreateStep]func(fromVertex *ictrltypes.LifecycleVertex, step pvcRecreateStep) *ictrltypes.LifecycleVertex{
- pvPolicyRetainStep: func(fromVertex *ictrltypes.LifecycleVertex, step pvcRecreateStep) *ictrltypes.LifecycleVertex {
- // step 1: update pv to retain
- retainPV := pv.DeepCopy()
- if retainPV.Labels == nil {
- retainPV.Labels = make(map[string]string)
- }
- // add label to pv, in case pvc get deleted, and we can't find pv
- retainPV.Labels[constant.PVCNameLabelKey] = pvcKey.Name
- if retainPV.Annotations == nil {
- retainPV.Annotations = make(map[string]string)
- }
- retainPV.Annotations[constant.PVLastClaimPolicyAnnotationKey] = string(pv.Spec.PersistentVolumeReclaimPolicy)
- retainPV.Spec.PersistentVolumeReclaimPolicy = corev1.PersistentVolumeReclaimRetain
- return c.PatchResource(retainPV, pv, fromVertex)
- },
- deletePVCStep: func(fromVertex *ictrltypes.LifecycleVertex, step pvcRecreateStep) *ictrltypes.LifecycleVertex {
- // step 2: delete pvc, this will not delete pv because policy is 'retain'
- removeFinalizerPVC := pvc.DeepCopy()
- removeFinalizerPVC.SetFinalizers([]string{})
- removeFinalizerPVCVertex := c.PatchResource(removeFinalizerPVC, pvc, fromVertex)
- return c.DeleteResource(pvc, removeFinalizerPVCVertex)
- },
- removePVClaimRefStep: func(fromVertex *ictrltypes.LifecycleVertex, step pvcRecreateStep) *ictrltypes.LifecycleVertex {
- // step 3: remove claimRef in pv
- removeClaimRefPV := pv.DeepCopy()
- if removeClaimRefPV.Spec.ClaimRef != nil {
- removeClaimRefPV.Spec.ClaimRef.UID = ""
- removeClaimRefPV.Spec.ClaimRef.ResourceVersion = ""
- }
- return c.PatchResource(removeClaimRefPV, pv, fromVertex)
- },
- createPVCStep: func(fromVertex *ictrltypes.LifecycleVertex, step pvcRecreateStep) *ictrltypes.LifecycleVertex {
- // step 4: create new pvc
- newPVC.SetResourceVersion("")
- return c.CreateResource(newPVC, fromVertex)
- },
- pvRestorePolicyStep: func(fromVertex *ictrltypes.LifecycleVertex, step pvcRecreateStep) *ictrltypes.LifecycleVertex {
- // step 5: restore to previous pv policy
- restorePV := pv.DeepCopy()
- policy := corev1.PersistentVolumeReclaimPolicy(restorePV.Annotations[constant.PVLastClaimPolicyAnnotationKey])
- if len(policy) == 0 {
- policy = corev1.PersistentVolumeReclaimDelete
- }
- restorePV.Spec.PersistentVolumeReclaimPolicy = policy
- return c.PatchResource(restorePV, pv, fromVertex)
- },
- }
-
- updatePVCByRecreateFromStep := func(fromStep pvcRecreateStep) {
- lastVertex := c.WorkloadVertex
- for step := pvRestorePolicyStep; step >= fromStep && step >= pvPolicyRetainStep; step-- {
- lastVertex = addStepMap[step](lastVertex, step)
- }
- }
-
- targetQuantity := vctProto.Spec.Resources.Requests[corev1.ResourceStorage]
- if pvcNotFound && !pvNotFound {
- // this could happen if create pvc step failed when recreating pvc
- updatePVCByRecreateFromStep(removePVClaimRefStep)
- return nil
- }
- if pvcNotFound && pvNotFound {
- // if both pvc and pv not found, do nothing
- return nil
- }
- if reflect.DeepEqual(pvc.Spec.Resources, newPVC.Spec.Resources) && pv.Spec.PersistentVolumeReclaimPolicy == corev1.PersistentVolumeReclaimRetain {
- // this could happen if create pvc succeeded but last step failed
- updatePVCByRecreateFromStep(pvRestorePolicyStep)
- return nil
- }
- if pvcQuantity := pvc.Spec.Resources.Requests[corev1.ResourceStorage]; !viper.GetBool(constant.CfgRecoverVolumeExpansionFailure) &&
- pvcQuantity.Cmp(targetQuantity) == 1 && // check if it's compressing volume
- targetQuantity.Cmp(*pvc.Status.Capacity.Storage()) >= 0 { // check if target size is greater than or equal to actual size
- // this branch means we can update pvc size by recreate it
- updatePVCByRecreateFromStep(pvPolicyRetainStep)
- return nil
- }
- if pvcQuantity := pvc.Spec.Resources.Requests[corev1.ResourceStorage]; pvcQuantity.Cmp(vctProto.Spec.Resources.Requests[corev1.ResourceStorage]) != 0 {
- // use pvc's update without anything extra
- c.UpdateResource(newPVC, c.WorkloadVertex)
- return nil
- }
- // all the else means no need to update
-
- return nil
-}
-
-func (c *rsmComponentBase) isAllConfigSynced(reqCtx intctrlutil.RequestCtx, cli client.Client) bool {
- checkFinishedReconfigure := func(cm *corev1.ConfigMap) bool {
- labels := cm.GetLabels()
- annotations := cm.GetAnnotations()
- if len(annotations) == 0 || len(labels) == 0 {
- return false
- }
- hash, _ := util.ComputeHash(cm.Data)
- return labels[constant.CMInsConfigurationHashLabelKey] == hash
- }
-
- var (
- cmKey client.ObjectKey
- cmObj = &corev1.ConfigMap{}
- allConfigSynced = true
- )
- for _, configSpec := range c.Component.ConfigTemplates {
- cmKey = client.ObjectKey{
- Namespace: c.GetNamespace(),
- Name: cfgcore.GetComponentCfgName(c.GetClusterName(), c.GetName(), configSpec.Name),
- }
- if err := cli.Get(reqCtx.Ctx, cmKey, cmObj); err != nil {
- return true
- }
- if !checkFinishedReconfigure(cmObj) {
- allConfigSynced = false
- break
- }
- }
- return allConfigSynced
-}
-
-func (c *rsmComponentBase) hasVolumeExpansionRunning(reqCtx intctrlutil.RequestCtx, cli client.Client) (bool, bool, error) {
- var (
- running bool
- failed bool
- )
- for _, vct := range c.runningWorkload.Spec.VolumeClaimTemplates {
- volumes, err := c.getRunningVolumes(reqCtx, cli, vct.Name, c.runningWorkload)
- if err != nil {
- return false, false, err
- }
- for _, v := range volumes {
- if v.Status.Capacity == nil || v.Status.Capacity.Storage().Cmp(v.Spec.Resources.Requests[corev1.ResourceStorage]) >= 0 {
- continue
- }
- running = true
- // TODO: how to check the expansion failed?
- }
- }
- return running, failed, nil
-}
-
-func (c *rsmComponentBase) HorizontalScale(reqCtx intctrlutil.RequestCtx, cli client.Client) error {
- return c.horizontalScale(reqCtx, cli)
-}
-
-func (c *rsmComponentBase) horizontalScale(reqCtx intctrlutil.RequestCtx, cli client.Client) error {
- sts := ConvertRSMToSTS(c.runningWorkload)
- if sts.Status.ReadyReplicas == c.Component.Replicas {
- return nil
- }
- ret := c.horizontalScaling(sts)
- if ret == 0 {
- if err := c.postScaleIn(reqCtx, cli); err != nil {
- return err
- }
- if err := c.postScaleOut(reqCtx, cli, sts); err != nil {
- return err
- }
- return nil
- }
- if ret < 0 {
- if err := c.scaleIn(reqCtx, cli, sts); err != nil {
- return err
- }
- } else {
- if err := c.scaleOut(reqCtx, cli, sts); err != nil {
- return err
- }
- }
-
- if err := c.updatePodReplicaLabel4Scaling(reqCtx, cli, c.Component.Replicas); err != nil {
- return err
- }
-
- // update KB___ env needed by pod to obtain hostname.
- c.updatePodEnvConfig()
-
- reqCtx.Recorder.Eventf(c.Cluster,
- corev1.EventTypeNormal,
- "HorizontalScale",
- "start horizontal scale component %s of cluster %s from %d to %d",
- c.GetName(), c.GetClusterName(), int(c.Component.Replicas)-ret, c.Component.Replicas)
-
- return nil
-}
-
-// < 0 for scale in, > 0 for scale out, and == 0 for nothing
-func (c *rsmComponentBase) horizontalScaling(stsObj *appsv1.StatefulSet) int {
- return int(c.Component.Replicas - *stsObj.Spec.Replicas)
-}
-
-func (c *rsmComponentBase) updatePodEnvConfig() {
- for _, v := range ictrltypes.FindAll[*corev1.ConfigMap](c.Dag) {
- node := v.(*ictrltypes.LifecycleVertex)
- // TODO: need a way to reference the env config.
- envConfigName := fmt.Sprintf("%s-%s-env", c.GetClusterName(), c.GetName())
- if node.Obj.GetName() == envConfigName {
- node.Action = ictrltypes.ActionUpdatePtr()
- }
- }
-}
-
-func (c *rsmComponentBase) updatePodReplicaLabel4Scaling(reqCtx intctrlutil.RequestCtx, cli client.Client, replicas int32) error {
- pods, err := listPodOwnedByComponent(reqCtx.Ctx, cli, c.GetNamespace(), c.GetMatchingLabels())
- if err != nil {
- return err
- }
- for _, pod := range pods {
- obj := pod.DeepCopy()
- if obj.Annotations == nil {
- obj.Annotations = make(map[string]string)
- }
- obj.Annotations[constant.ComponentReplicasAnnotationKey] = strconv.Itoa(int(replicas))
- c.UpdateResource(obj, c.WorkloadVertex)
- }
- return nil
-}
-
-func (c *rsmComponentBase) scaleIn(reqCtx intctrlutil.RequestCtx, cli client.Client, stsObj *appsv1.StatefulSet) error {
- // if scale in to 0, do not delete pvcs
- if c.Component.Replicas == 0 {
- reqCtx.Log.Info("scale in to 0, keep all PVCs")
- return nil
- }
- // TODO: check the component definition to determine whether we need to call leave member before deleting replicas.
- err := c.leaveMember4ScaleIn(reqCtx, cli, stsObj)
- if err != nil {
- reqCtx.Log.Info(fmt.Sprintf("leave member at scaling-in error, retry later: %s", err.Error()))
- return err
- }
- return c.deletePVCs4ScaleIn(reqCtx, cli, stsObj)
-}
-
-func (c *rsmComponentBase) postScaleIn(reqCtx intctrlutil.RequestCtx, cli client.Client) error {
- return nil
-}
-
-func (c *rsmComponentBase) leaveMember4ScaleIn(reqCtx intctrlutil.RequestCtx, cli client.Client, stsObj *appsv1.StatefulSet) error {
- pods, err := listPodOwnedByComponent(reqCtx.Ctx, cli, c.GetNamespace(), c.GetMatchingLabels())
- if err != nil {
- return err
- }
- for _, pod := range pods {
- subs := strings.Split(pod.Name, "-")
- if ordinal, err := strconv.ParseInt(subs[len(subs)-1], 10, 32); err != nil {
- return err
- } else if int32(ordinal) < c.Component.Replicas {
- continue
- }
- lorryCli, err1 := lorry.NewClient(c.Component.CharacterType, *pod)
- if err1 != nil {
- if err == nil {
- err = err1
- }
- continue
- }
- if err2 := lorryCli.LeaveMember(reqCtx.Ctx); err2 != nil {
- if err == nil {
- err = err2
- }
- }
- }
- return err // TODO: use requeue-after
-}
-
-func (c *rsmComponentBase) deletePVCs4ScaleIn(reqCtx intctrlutil.RequestCtx, cli client.Client, stsObj *appsv1.StatefulSet) error {
- for i := c.Component.Replicas; i < *stsObj.Spec.Replicas; i++ {
- for _, vct := range stsObj.Spec.VolumeClaimTemplates {
- pvcKey := types.NamespacedName{
- Namespace: stsObj.Namespace,
- Name: fmt.Sprintf("%s-%s-%d", vct.Name, stsObj.Name, i),
- }
- pvc := corev1.PersistentVolumeClaim{}
- if err := cli.Get(reqCtx.Ctx, pvcKey, &pvc); err != nil {
- return err
- }
- // Since there are no order guarantee between updating STS and deleting PVCs, if there is any error occurred
- // after updating STS and before deleting PVCs, the PVCs intended to scale-in will be leaked.
- // For simplicity, the updating dependency is added between them to guarantee that the PVCs to scale-in
- // will be deleted or the scaling-in operation will be failed.
- c.DeleteResource(&pvc, c.WorkloadVertex)
- }
- }
- return nil
-}
-
-func (c *rsmComponentBase) scaleOut(reqCtx intctrlutil.RequestCtx, cli client.Client, stsObj *appsv1.StatefulSet) error {
- var (
- backupKey = types.NamespacedName{
- Namespace: stsObj.Namespace,
- Name: stsObj.Name + "-scaling",
- }
- )
-
- // sts's replicas=0 means it's starting not scaling, skip all the scaling work.
- if *stsObj.Spec.Replicas == 0 {
- return nil
- }
-
- c.WorkloadVertex.Immutable = true
- rsmProto := c.WorkloadVertex.Obj.(*workloads.ReplicatedStateMachine)
- stsProto := ConvertRSMToSTS(rsmProto)
- d, err := newDataClone(reqCtx, cli, c.Cluster, c.Component, stsObj, stsProto, backupKey)
- if err != nil {
- return err
- }
- var succeed bool
- if d == nil {
- succeed = true
- } else {
- succeed, err = d.succeed()
- if err != nil {
- return err
- }
- }
- if succeed {
- // pvcs are ready, rsm.replicas should be updated
- c.WorkloadVertex.Immutable = false
- return c.postScaleOut(reqCtx, cli, stsObj)
- } else {
- c.WorkloadVertex.Immutable = true
- // update objs will trigger cluster reconcile, no need to requeue error
- objs, err := d.cloneData(d)
- if err != nil {
- return err
- }
- for _, obj := range objs {
- c.CreateResource(obj, nil)
- }
- return nil
- }
-}
-
-func (c *rsmComponentBase) postScaleOut(reqCtx intctrlutil.RequestCtx, cli client.Client, stsObj *appsv1.StatefulSet) error {
- var (
- snapshotKey = types.NamespacedName{
- Namespace: stsObj.Namespace,
- Name: stsObj.Name + "-scaling",
- }
- )
-
- d, err := newDataClone(reqCtx, cli, c.Cluster, c.Component, stsObj, stsObj, snapshotKey)
- if err != nil {
- return err
- }
- if d != nil {
- // clean backup resources.
- // there will not be any backup resources other than scale out.
- tmpObjs, err := d.clearTmpResources()
- if err != nil {
- return err
- }
- for _, obj := range tmpObjs {
- c.DeleteResource(obj, nil)
- }
- }
-
- return nil
-}
-
-func (c *rsmComponentBase) updateUnderlyingResources(reqCtx intctrlutil.RequestCtx, cli client.Client, rsmObj *workloads.ReplicatedStateMachine) error {
- if rsmObj == nil {
- c.createWorkload()
- } else {
- c.updateWorkload(rsmObj)
- // to work around that the scaled PVC will be deleted at object action.
- if err := c.updateVolumes(reqCtx, cli, rsmObj); err != nil {
- return err
- }
- }
- if err := c.UpdatePDB(reqCtx, cli); err != nil {
- return err
- }
- return nil
-}
-
-func (c *rsmComponentBase) createWorkload() {
- rsmProto := c.WorkloadVertex.Obj.(*workloads.ReplicatedStateMachine)
- buildWorkLoadAnnotations(rsmProto, c.Cluster)
- c.WorkloadVertex.Obj = rsmProto
- c.WorkloadVertex.Action = ictrltypes.ActionCreatePtr()
-}
-
-func (c *rsmComponentBase) updateWorkload(rsmObj *workloads.ReplicatedStateMachine) bool {
- rsmObjCopy := rsmObj.DeepCopy()
- rsmProto := c.WorkloadVertex.Obj.(*workloads.ReplicatedStateMachine)
-
- // remove original monitor annotations
- if len(rsmObjCopy.Annotations) > 0 {
- maps.DeleteFunc(rsmObjCopy.Annotations, func(k, v string) bool {
- return strings.HasPrefix(k, "monitor.kubeblocks.io")
- })
- }
- mergeAnnotations(rsmObjCopy.Annotations, &rsmProto.Annotations)
- rsmObjCopy.Annotations = rsmProto.Annotations
- buildWorkLoadAnnotations(rsmObjCopy, c.Cluster)
-
- // keep the original template annotations.
- // if annotations exist and are replaced, the rsm will be updated.
- mergeAnnotations(rsmObjCopy.Spec.Template.Annotations, &rsmProto.Spec.Template.Annotations)
- rsmObjCopy.Spec.Template = rsmProto.Spec.Template
- rsmObjCopy.Spec.Replicas = rsmProto.Spec.Replicas
- c.updateUpdateStrategy(rsmObjCopy, rsmProto)
- rsmObjCopy.Spec.Service = rsmProto.Spec.Service
- rsmObjCopy.Spec.AlternativeServices = rsmProto.Spec.AlternativeServices
- rsmObjCopy.Spec.Roles = rsmProto.Spec.Roles
- rsmObjCopy.Spec.RoleProbe = rsmProto.Spec.RoleProbe
- rsmObjCopy.Spec.MembershipReconfiguration = rsmProto.Spec.MembershipReconfiguration
- rsmObjCopy.Spec.MemberUpdateStrategy = rsmProto.Spec.MemberUpdateStrategy
- rsmObjCopy.Spec.Credential = rsmProto.Spec.Credential
-
- resolvePodSpecDefaultFields(rsmObj.Spec.Template.Spec, &rsmObjCopy.Spec.Template.Spec)
-
- delayUpdatePodSpecSystemFields(rsmObj.Spec.Template.Spec, &rsmObjCopy.Spec.Template.Spec)
- isTemplateUpdated := !reflect.DeepEqual(&rsmObj.Spec, &rsmObjCopy.Spec)
- if isTemplateUpdated {
- updatePodSpecSystemFields(&rsmObjCopy.Spec.Template.Spec)
- }
- if isTemplateUpdated || !reflect.DeepEqual(rsmObj.Annotations, rsmObjCopy.Annotations) {
- c.WorkloadVertex.Obj = rsmObjCopy
- c.WorkloadVertex.Action = ictrltypes.ActionPtr(ictrltypes.UPDATE)
- return true
- }
- return false
-}
-
-func (c *rsmComponentBase) updateUpdateStrategy(rsmObj, rsmProto *workloads.ReplicatedStateMachine) {
- var objMaxUnavailable *intstr.IntOrString
- if rsmObj.Spec.UpdateStrategy.RollingUpdate != nil {
- objMaxUnavailable = rsmObj.Spec.UpdateStrategy.RollingUpdate.MaxUnavailable
- }
- rsmObj.Spec.UpdateStrategy = rsmProto.Spec.UpdateStrategy
- if objMaxUnavailable == nil && rsmObj.Spec.UpdateStrategy.RollingUpdate != nil {
- // HACK: This field is alpha-level (since v1.24) and is only honored by servers that enable the
- // MaxUnavailableStatefulSet feature.
- // When we get a nil MaxUnavailable from k8s, we consider that the field is not supported by the server,
- // and set the MaxUnavailable as nil explicitly to avoid the workload been updated unexpectedly.
- // Ref: https://kubernetes.io/docs/concepts/workloads/controllers/statefulset/#maximum-unavailable-pods
- rsmObj.Spec.UpdateStrategy.RollingUpdate.MaxUnavailable = nil
- }
-}
-
-func (c *rsmComponentBase) updateVolumes(reqCtx intctrlutil.RequestCtx, cli client.Client, rsmObj *workloads.ReplicatedStateMachine) error {
- // PVCs which have been added to the dag because of volume expansion.
- pvcNameSet := sets.New[string]()
- for _, v := range ictrltypes.FindAll[*corev1.PersistentVolumeClaim](c.Dag) {
- pvcNameSet.Insert(v.(*ictrltypes.LifecycleVertex).Obj.GetName())
- }
-
- for _, vct := range c.Component.VolumeClaimTemplates {
- pvcs, err := c.getRunningVolumes(reqCtx, cli, vct.Name, rsmObj)
- if err != nil {
- return err
- }
- for _, pvc := range pvcs {
- if pvcNameSet.Has(pvc.Name) {
- continue
- }
- c.NoopResource(pvc, c.WorkloadVertex)
- }
- }
- return nil
-}
-
-func (c *rsmComponentBase) getRunningVolumes(reqCtx intctrlutil.RequestCtx, cli client.Client, vctName string,
- rsmObj *workloads.ReplicatedStateMachine) ([]*corev1.PersistentVolumeClaim, error) {
- pvcs, err := listObjWithLabelsInNamespace(reqCtx.Ctx, cli, generics.PersistentVolumeClaimSignature, c.GetNamespace(), c.GetMatchingLabels())
- if err != nil {
- if apierrors.IsNotFound(err) {
- return nil, nil
- }
- return nil, err
- }
- matchedPVCs := make([]*corev1.PersistentVolumeClaim, 0)
- prefix := fmt.Sprintf("%s-%s", vctName, rsmObj.Name)
- for _, pvc := range pvcs {
- if strings.HasPrefix(pvc.Name, prefix) {
- matchedPVCs = append(matchedPVCs, pvc)
- }
- }
- return matchedPVCs, nil
-}
diff --git a/controllers/apps/components/base_stateful_legacy.go b/controllers/apps/components/base_stateful_legacy.go
deleted file mode 100644
index 973778b44ff..00000000000
--- a/controllers/apps/components/base_stateful_legacy.go
+++ /dev/null
@@ -1,752 +0,0 @@
-/*
-Copyright (C) 2022-2023 ApeCloud Co., Ltd
-
-This file is part of KubeBlocks project
-
-This program is free software: you can redistribute it and/or modify
-it under the terms of the GNU Affero General Public License as published by
-the Free Software Foundation, either version 3 of the License, or
-(at your option) any later version.
-
-This program is distributed in the hope that it will be useful
-but WITHOUT ANY WARRANTY; without even the implied warranty of
-MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
-GNU Affero General Public License for more details.
-
-You should have received a copy of the GNU Affero General Public License
-along with this program. If not, see .
-*/
-
-package components
-
-import (
- "fmt"
- "reflect"
- "strconv"
- "strings"
-
- appsv1 "k8s.io/api/apps/v1"
- corev1 "k8s.io/api/core/v1"
- apierrors "k8s.io/apimachinery/pkg/api/errors"
- "k8s.io/apimachinery/pkg/types"
- "k8s.io/apimachinery/pkg/util/intstr"
- "k8s.io/apimachinery/pkg/util/sets"
- "sigs.k8s.io/controller-runtime/pkg/client"
-
- appsv1alpha1 "github.com/apecloud/kubeblocks/apis/apps/v1alpha1"
- "github.com/apecloud/kubeblocks/internal/constant"
- "github.com/apecloud/kubeblocks/internal/controller/graph"
- ictrltypes "github.com/apecloud/kubeblocks/internal/controller/types"
- intctrlutil "github.com/apecloud/kubeblocks/internal/controllerutil"
- "github.com/apecloud/kubeblocks/internal/generics"
- viper "github.com/apecloud/kubeblocks/internal/viperx"
-)
-
-// statefulComponentBase as a base class for single stateful-set based component (stateful & replication & consensus).
-type statefulComponentBase struct {
- componentBase
- // runningWorkload can be nil, and the replicas of workload can be nil (zero)
- runningWorkload *appsv1.StatefulSet
-}
-
-func (c *statefulComponentBase) init(reqCtx intctrlutil.RequestCtx, cli client.Client, builder componentWorkloadBuilder, load bool) error {
- var err error
- if builder != nil {
- if err = builder.BuildEnv().
- BuildWorkload().
- BuildPDB().
- BuildHeadlessService().
- BuildConfig().
- BuildTLSVolume().
- BuildVolumeMount().
- BuildService().
- BuildTLSCert().
- Complete(); err != nil {
- return err
- }
- }
- if load {
- c.runningWorkload, err = c.loadRunningWorkload(reqCtx, cli)
- if err != nil {
- return err
- }
- }
- return nil
-}
-
-func (c *statefulComponentBase) loadRunningWorkload(reqCtx intctrlutil.RequestCtx, cli client.Client) (*appsv1.StatefulSet, error) {
- stsList, err := listStsOwnedByComponent(reqCtx.Ctx, cli, c.GetNamespace(), c.GetMatchingLabels())
- if err != nil {
- return nil, err
- }
- cnt := len(stsList)
- if cnt == 1 {
- return stsList[0], nil
- }
- if cnt == 0 {
- return nil, nil
- } else {
- return nil, fmt.Errorf("more than one workloads found for the component, cluster: %s, component: %s, cnt: %d",
- c.GetClusterName(), c.GetName(), cnt)
- }
-}
-
-func (c *statefulComponentBase) GetBuiltObjects(builder componentWorkloadBuilder) ([]client.Object, error) {
- dag := c.Dag
- defer func() {
- c.Dag = dag
- }()
-
- c.Dag = graph.NewDAG()
- if err := c.init(intctrlutil.RequestCtx{}, nil, builder, false); err != nil {
- return nil, err
- }
-
- objs := make([]client.Object, 0)
- for _, v := range c.Dag.Vertices() {
- if vv, ok := v.(*ictrltypes.LifecycleVertex); ok {
- objs = append(objs, vv.Obj)
- }
- }
- return objs, nil
-}
-
-func (c *statefulComponentBase) Create(reqCtx intctrlutil.RequestCtx, cli client.Client, builder componentWorkloadBuilder) error {
- if err := c.init(reqCtx, cli, builder, false); err != nil {
- return err
- }
-
- if err := c.ValidateObjectsAction(); err != nil {
- return err
- }
-
- c.SetStatusPhase(appsv1alpha1.CreatingClusterCompPhase, nil, "Create a new component")
-
- return nil
-}
-
-func (c *statefulComponentBase) Delete(reqCtx intctrlutil.RequestCtx, cli client.Client) error {
- // TODO(impl): delete component owned resources
- return nil
-}
-
-func (c *statefulComponentBase) Update(reqCtx intctrlutil.RequestCtx, cli client.Client, builder componentWorkloadBuilder) error {
- if err := c.init(reqCtx, cli, builder, true); err != nil {
- return err
- }
-
- if c.runningWorkload != nil {
- if err := c.Restart(reqCtx, cli); err != nil {
- return err
- }
-
- // cluster.spec.componentSpecs[*].volumeClaimTemplates[*].spec.resources.requests[corev1.ResourceStorage]
- if err := c.ExpandVolume(reqCtx, cli); err != nil {
- return err
- }
-
- // cluster.spec.componentSpecs[*].replicas
- if err := c.HorizontalScale(reqCtx, cli); err != nil {
- return err
- }
- }
-
- if err := c.updateUnderlyingResources(reqCtx, cli, c.runningWorkload); err != nil {
- return err
- }
-
- return c.ResolveObjectsAction(reqCtx, cli)
-}
-
-func (c *statefulComponentBase) Status(reqCtx intctrlutil.RequestCtx, cli client.Client, builder componentWorkloadBuilder) error {
- if err := c.init(reqCtx, cli, builder, true); err != nil {
- return err
- }
- if c.runningWorkload == nil {
- return nil
- }
-
- statusTxn := &statusReconciliationTxn{}
-
- if err := c.statusExpandVolume(reqCtx, cli, statusTxn); err != nil {
- return err
- }
-
- if err := c.horizontalScale(reqCtx, cli, statusTxn); err != nil {
- return err
- }
-
- if vertexes, err := c.ComponentSet.HandleRoleChange(reqCtx.Ctx, c.runningWorkload); err != nil {
- return err
- } else {
- for _, v := range vertexes {
- c.Dag.AddVertex(v)
- }
- }
-
- // TODO(impl): restart pod if needed, move it to @Update and restart pod directly.
- if vertexes, err := c.ComponentSet.HandleRestart(reqCtx.Ctx, c.runningWorkload); err != nil {
- return err
- } else {
- for _, v := range vertexes {
- c.Dag.AddVertex(v)
- }
- }
-
- var delayedRequeueError error
- if err := c.StatusWorkload(reqCtx, cli, c.runningWorkload, statusTxn); err != nil {
- if !intctrlutil.IsDelayedRequeueError(err) {
- return err
- }
- delayedRequeueError = err
- }
-
- if err := statusTxn.commit(); err != nil {
- return err
- }
-
- c.updateWorkload(c.runningWorkload)
-
- // update component info to pods' annotations
- if err := updateComponentInfoToPods(reqCtx.Ctx, cli, c.Cluster, c.Component, c.Dag); err != nil {
- return err
- }
-
- // patch the current componentSpec workload's custom labels
- if err := updateCustomLabelToPods(reqCtx.Ctx, cli, c.Cluster, c.Component, c.Dag); err != nil {
- reqCtx.Event(c.Cluster, corev1.EventTypeWarning, "Component Workload Controller PatchWorkloadCustomLabelFailed", err.Error())
- return err
- }
-
- return delayedRequeueError
-}
-
-func (c *statefulComponentBase) Restart(reqCtx intctrlutil.RequestCtx, cli client.Client) error {
- return restartPod(&c.runningWorkload.Spec.Template)
-}
-
-func (c *statefulComponentBase) ExpandVolume(reqCtx intctrlutil.RequestCtx, cli client.Client) error {
- for _, vct := range c.runningWorkload.Spec.VolumeClaimTemplates {
- var proto *corev1.PersistentVolumeClaimTemplate
- for _, v := range c.Component.VolumeClaimTemplates {
- if v.Name == vct.Name {
- proto = &v
- break
- }
- }
- // REVIEW: seems we can remove a volume claim from templates at runtime, without any changes and warning messages?
- if proto == nil {
- continue
- }
-
- if err := c.expandVolumes(reqCtx, cli, vct.Name, proto); err != nil {
- return err
- }
- }
- return nil
-}
-
-func (c *statefulComponentBase) expandVolumes(reqCtx intctrlutil.RequestCtx, cli client.Client,
- vctName string, proto *corev1.PersistentVolumeClaimTemplate) error {
- pvcNotFound := false
- for i := *c.runningWorkload.Spec.Replicas - 1; i >= 0; i-- {
- pvc := &corev1.PersistentVolumeClaim{}
- pvcKey := types.NamespacedName{
- Namespace: c.GetNamespace(),
- Name: fmt.Sprintf("%s-%s-%d", vctName, c.runningWorkload.Name, i),
- }
- if err := cli.Get(reqCtx.Ctx, pvcKey, pvc); err != nil {
- if apierrors.IsNotFound(err) {
- pvcNotFound = true
- } else {
- return err
- }
- }
- if err := c.updatePVCSize(reqCtx, cli, pvcKey, pvc, pvcNotFound, proto); err != nil {
- return err
- }
- }
- return nil
-}
-
-func (c *statefulComponentBase) updatePVCSize(reqCtx intctrlutil.RequestCtx, cli client.Client, pvcKey types.NamespacedName,
- pvc *corev1.PersistentVolumeClaim, pvcNotFound bool, vctProto *corev1.PersistentVolumeClaimTemplate) error {
- // reference: https://kubernetes.io/docs/concepts/storage/persistent-volumes/#recovering-from-failure-when-expanding-volumes
- // 1. Mark the PersistentVolume(PV) that is bound to the PersistentVolumeClaim(PVC) with Retain reclaim policy.
- // 2. Delete the PVC. Since PV has Retain reclaim policy - we will not lose any data when we recreate the PVC.
- // 3. Delete the claimRef entry from PV specs, so as new PVC can bind to it. This should make the PV Available.
- // 4. Re-create the PVC with smaller size than PV and set volumeName field of the PVC to the name of the PV. This should bind new PVC to existing PV.
- // 5. Don't forget to restore the reclaim policy of the PV.
- newPVC := pvc.DeepCopy()
- if pvcNotFound {
- newPVC.Name = pvcKey.Name
- newPVC.Namespace = pvcKey.Namespace
- newPVC.SetLabels(vctProto.Labels)
- newPVC.Spec = vctProto.Spec
- ml := client.MatchingLabels{
- constant.PVCNameLabelKey: pvcKey.Name,
- }
- pvList := corev1.PersistentVolumeList{}
- if err := cli.List(reqCtx.Ctx, &pvList, ml); err != nil {
- return err
- }
- for _, pv := range pvList.Items {
- // find pv referenced this pvc
- if pv.Spec.ClaimRef == nil {
- continue
- }
- if pv.Spec.ClaimRef.Name == pvcKey.Name {
- newPVC.Spec.VolumeName = pv.Name
- break
- }
- }
- } else {
- newPVC.Spec.Resources.Requests[corev1.ResourceStorage] = vctProto.Spec.Resources.Requests[corev1.ResourceStorage]
- // delete annotation to make it re-bind
- delete(newPVC.Annotations, "pv.kubernetes.io/bind-completed")
- }
-
- pvNotFound := false
-
- // step 1: update pv to retain
- pv := &corev1.PersistentVolume{}
- pvKey := types.NamespacedName{
- Namespace: pvcKey.Namespace,
- Name: newPVC.Spec.VolumeName,
- }
- if err := cli.Get(reqCtx.Ctx, pvKey, pv); err != nil {
- if apierrors.IsNotFound(err) {
- pvNotFound = true
- } else {
- return err
- }
- }
-
- type pvcRecreateStep int
- const (
- pvPolicyRetainStep pvcRecreateStep = iota
- deletePVCStep
- removePVClaimRefStep
- createPVCStep
- pvRestorePolicyStep
- )
-
- addStepMap := map[pvcRecreateStep]func(fromVertex *ictrltypes.LifecycleVertex, step pvcRecreateStep) *ictrltypes.LifecycleVertex{
- pvPolicyRetainStep: func(fromVertex *ictrltypes.LifecycleVertex, step pvcRecreateStep) *ictrltypes.LifecycleVertex {
- // step 1: update pv to retain
- retainPV := pv.DeepCopy()
- if retainPV.Labels == nil {
- retainPV.Labels = make(map[string]string)
- }
- // add label to pv, in case pvc get deleted, and we can't find pv
- retainPV.Labels[constant.PVCNameLabelKey] = pvcKey.Name
- if retainPV.Annotations == nil {
- retainPV.Annotations = make(map[string]string)
- }
- retainPV.Annotations[constant.PVLastClaimPolicyAnnotationKey] = string(pv.Spec.PersistentVolumeReclaimPolicy)
- retainPV.Spec.PersistentVolumeReclaimPolicy = corev1.PersistentVolumeReclaimRetain
- return c.PatchResource(retainPV, pv, fromVertex)
- },
- deletePVCStep: func(fromVertex *ictrltypes.LifecycleVertex, step pvcRecreateStep) *ictrltypes.LifecycleVertex {
- // step 2: delete pvc, this will not delete pv because policy is 'retain'
- removeFinalizerPVC := pvc.DeepCopy()
- removeFinalizerPVC.SetFinalizers([]string{})
- removeFinalizerPVCVertex := c.PatchResource(removeFinalizerPVC, pvc, fromVertex)
- return c.DeleteResource(pvc, removeFinalizerPVCVertex)
- },
- removePVClaimRefStep: func(fromVertex *ictrltypes.LifecycleVertex, step pvcRecreateStep) *ictrltypes.LifecycleVertex {
- // step 3: remove claimRef in pv
- removeClaimRefPV := pv.DeepCopy()
- if removeClaimRefPV.Spec.ClaimRef != nil {
- removeClaimRefPV.Spec.ClaimRef.UID = ""
- removeClaimRefPV.Spec.ClaimRef.ResourceVersion = ""
- }
- return c.PatchResource(removeClaimRefPV, pv, fromVertex)
- },
- createPVCStep: func(fromVertex *ictrltypes.LifecycleVertex, step pvcRecreateStep) *ictrltypes.LifecycleVertex {
- // step 4: create new pvc
- newPVC.SetResourceVersion("")
- return c.CreateResource(newPVC, fromVertex)
- },
- pvRestorePolicyStep: func(fromVertex *ictrltypes.LifecycleVertex, step pvcRecreateStep) *ictrltypes.LifecycleVertex {
- // step 5: restore to previous pv policy
- restorePV := pv.DeepCopy()
- policy := corev1.PersistentVolumeReclaimPolicy(restorePV.Annotations[constant.PVLastClaimPolicyAnnotationKey])
- if len(policy) == 0 {
- policy = corev1.PersistentVolumeReclaimDelete
- }
- restorePV.Spec.PersistentVolumeReclaimPolicy = policy
- return c.PatchResource(restorePV, pv, fromVertex)
- },
- }
-
- updatePVCByRecreateFromStep := func(fromStep pvcRecreateStep) {
- lastVertex := c.WorkloadVertex
- for step := pvRestorePolicyStep; step >= fromStep && step >= pvPolicyRetainStep; step-- {
- lastVertex = addStepMap[step](lastVertex, step)
- }
- }
-
- targetQuantity := vctProto.Spec.Resources.Requests[corev1.ResourceStorage]
- if pvcNotFound && !pvNotFound {
- // this could happen if create pvc step failed when recreating pvc
- updatePVCByRecreateFromStep(removePVClaimRefStep)
- return nil
- }
- if pvcNotFound && pvNotFound {
- // if both pvc and pv not found, do nothing
- return nil
- }
- if reflect.DeepEqual(pvc.Spec.Resources, newPVC.Spec.Resources) && pv.Spec.PersistentVolumeReclaimPolicy == corev1.PersistentVolumeReclaimRetain {
- // this could happen if create pvc succeeded but last step failed
- updatePVCByRecreateFromStep(pvRestorePolicyStep)
- return nil
- }
- if pvcQuantity := pvc.Spec.Resources.Requests[corev1.ResourceStorage]; !viper.GetBool(constant.CfgRecoverVolumeExpansionFailure) &&
- pvcQuantity.Cmp(targetQuantity) == 1 && // check if it's compressing volume
- targetQuantity.Cmp(*pvc.Status.Capacity.Storage()) >= 0 { // check if target size is greater than or equal to actual size
- // this branch means we can update pvc size by recreate it
- updatePVCByRecreateFromStep(pvPolicyRetainStep)
- return nil
- }
- if pvcQuantity := pvc.Spec.Resources.Requests[corev1.ResourceStorage]; pvcQuantity.Cmp(vctProto.Spec.Resources.Requests[corev1.ResourceStorage]) != 0 {
- // use pvc's update without anything extra
- c.UpdateResource(newPVC, c.WorkloadVertex)
- return nil
- }
- // all the else means no need to update
-
- return nil
-}
-
-func (c *statefulComponentBase) statusExpandVolume(reqCtx intctrlutil.RequestCtx, cli client.Client, txn *statusReconciliationTxn) error {
- for _, vct := range c.runningWorkload.Spec.VolumeClaimTemplates {
- running, failed, err := c.hasVolumeExpansionRunning(reqCtx, cli, vct.Name)
- if err != nil {
- return err
- }
- if failed {
- txn.propose(appsv1alpha1.AbnormalClusterCompPhase, func() {
- c.SetStatusPhase(appsv1alpha1.AbnormalClusterCompPhase, nil, "Volume Expansion failed")
- })
- return nil
- }
- if running {
- txn.propose(appsv1alpha1.UpdatingClusterCompPhase, func() {
- c.SetStatusPhase(appsv1alpha1.UpdatingClusterCompPhase, nil, "Volume Expansion failed")
- })
- return nil
- }
- }
- return nil
-}
-
-func (c *statefulComponentBase) hasVolumeExpansionRunning(reqCtx intctrlutil.RequestCtx, cli client.Client, vctName string) (bool, bool, error) {
- var (
- running bool
- failed bool
- )
- volumes, err := c.getRunningVolumes(reqCtx, cli, vctName, c.runningWorkload)
- if err != nil {
- return false, false, err
- }
- for _, v := range volumes {
- if v.Status.Capacity == nil || v.Status.Capacity.Storage().Cmp(v.Spec.Resources.Requests[corev1.ResourceStorage]) >= 0 {
- continue
- }
- running = true
- // TODO: how to check the expansion failed?
- }
- return running, failed, nil
-}
-
-func (c *statefulComponentBase) HorizontalScale(reqCtx intctrlutil.RequestCtx, cli client.Client) error {
- return c.horizontalScale(reqCtx, cli, nil)
-}
-
-func (c *statefulComponentBase) horizontalScale(reqCtx intctrlutil.RequestCtx, cli client.Client, txn *statusReconciliationTxn) error {
- sts := c.runningWorkload
- if sts.Status.ReadyReplicas == c.Component.Replicas {
- return nil
- }
- ret := c.horizontalScaling(sts)
- if ret == 0 {
- if err := c.postScaleIn(reqCtx, cli, txn); err != nil {
- return err
- }
- if err := c.postScaleOut(reqCtx, cli, sts); err != nil {
- return err
- }
- return nil
- }
- if ret < 0 {
- if err := c.scaleIn(reqCtx, cli, sts); err != nil {
- return err
- }
- } else {
- if err := c.scaleOut(reqCtx, cli, sts); err != nil {
- return err
- }
- }
-
- if err := c.updatePodReplicaLabel4Scaling(reqCtx, cli, c.Component.Replicas); err != nil {
- return err
- }
-
- // update KB___ env needed by pod to obtain hostname.
- c.updatePodEnvConfig()
-
- reqCtx.Recorder.Eventf(c.Cluster,
- corev1.EventTypeNormal,
- "HorizontalScale",
- "start horizontal scale component %s of cluster %s from %d to %d",
- c.GetName(), c.GetClusterName(), int(c.Component.Replicas)-ret, c.Component.Replicas)
-
- return nil
-}
-
-// < 0 for scale in, > 0 for scale out, and == 0 for nothing
-func (c *statefulComponentBase) horizontalScaling(stsObj *appsv1.StatefulSet) int {
- return int(c.Component.Replicas - *stsObj.Spec.Replicas)
-}
-
-func (c *statefulComponentBase) updatePodEnvConfig() {
- for _, v := range ictrltypes.FindAll[*corev1.ConfigMap](c.Dag) {
- node := v.(*ictrltypes.LifecycleVertex)
- // TODO: need a way to reference the env config.
- envConfigName := fmt.Sprintf("%s-%s-env", c.GetClusterName(), c.GetName())
- if node.Obj.GetName() == envConfigName {
- node.Action = ictrltypes.ActionUpdatePtr()
- }
- }
-}
-
-func (c *statefulComponentBase) updatePodReplicaLabel4Scaling(reqCtx intctrlutil.RequestCtx, cli client.Client, replicas int32) error {
- pods, err := listPodOwnedByComponent(reqCtx.Ctx, cli, c.GetNamespace(), c.GetMatchingLabels())
- if err != nil {
- return err
- }
- for _, pod := range pods {
- obj := pod.DeepCopy()
- if obj.Annotations == nil {
- obj.Annotations = make(map[string]string)
- }
- obj.Annotations[constant.ComponentReplicasAnnotationKey] = strconv.Itoa(int(replicas))
- c.UpdateResource(obj, c.WorkloadVertex)
- }
- return nil
-}
-
-func (c *statefulComponentBase) scaleIn(reqCtx intctrlutil.RequestCtx, cli client.Client, stsObj *appsv1.StatefulSet) error {
- // if scale in to 0, do not delete pvcs
- if c.Component.Replicas == 0 {
- return nil
- }
- for i := c.Component.Replicas; i < *stsObj.Spec.Replicas; i++ {
- for _, vct := range stsObj.Spec.VolumeClaimTemplates {
- pvcKey := types.NamespacedName{
- Namespace: stsObj.Namespace,
- Name: fmt.Sprintf("%s-%s-%d", vct.Name, stsObj.Name, i),
- }
- pvc := corev1.PersistentVolumeClaim{}
- if err := cli.Get(reqCtx.Ctx, pvcKey, &pvc); err != nil {
- return err
- }
- // Since there are no order guarantee between updating STS and deleting PVCs, if there is any error occurred
- // after updating STS and before deleting PVCs, the PVCs intended to scale-in will be leaked.
- // For simplicity, the updating dependency is added between them to guarantee that the PVCs to scale-in
- // will be deleted or the scaling-in operation will be failed.
- c.DeleteResource(&pvc, c.WorkloadVertex)
- }
- }
- return nil
-}
-
-func (c *statefulComponentBase) postScaleIn(reqCtx intctrlutil.RequestCtx, cli client.Client, txn *statusReconciliationTxn) error {
- return nil
-}
-
-func (c *statefulComponentBase) scaleOut(reqCtx intctrlutil.RequestCtx, cli client.Client, stsObj *appsv1.StatefulSet) error {
- var (
- backupKey = types.NamespacedName{
- Namespace: stsObj.Namespace,
- Name: stsObj.Name + "-scaling",
- }
- )
-
- // sts's replicas=0 means it's starting not scaling, skip all the scaling work.
- if *stsObj.Spec.Replicas == 0 {
- return nil
- }
-
- c.WorkloadVertex.Immutable = true
- stsProto := c.WorkloadVertex.Obj.(*appsv1.StatefulSet)
- d, err := newDataClone(reqCtx, cli, c.Cluster, c.Component, stsObj, stsProto, backupKey)
- if err != nil {
- return err
- }
- var succeed bool
- if d == nil {
- succeed = true
- } else {
- succeed, err = d.succeed()
- if err != nil {
- return err
- }
- }
- if succeed {
- // pvcs are ready, stateful_set.replicas should be updated
- c.WorkloadVertex.Immutable = false
- return c.postScaleOut(reqCtx, cli, stsObj)
- } else {
- c.WorkloadVertex.Immutable = true
- // update objs will trigger cluster reconcile, no need to requeue error
- objs, err := d.cloneData(d)
- if err != nil {
- return err
- }
- for _, obj := range objs {
- c.CreateResource(obj, nil)
- }
- return nil
- }
-}
-
-func (c *statefulComponentBase) postScaleOut(reqCtx intctrlutil.RequestCtx, cli client.Client, stsObj *appsv1.StatefulSet) error {
- var (
- snapshotKey = types.NamespacedName{
- Namespace: stsObj.Namespace,
- Name: stsObj.Name + "-scaling",
- }
- )
-
- d, err := newDataClone(reqCtx, cli, c.Cluster, c.Component, stsObj, stsObj, snapshotKey)
- if err != nil {
- return err
- }
- if d != nil {
- // clean backup resources.
- // there will not be any backup resources other than scale out.
- tmpObjs, err := d.clearTmpResources()
- if err != nil {
- return err
- }
- for _, obj := range tmpObjs {
- c.DeleteResource(obj, nil)
- }
- }
-
- return nil
-}
-
-func (c *statefulComponentBase) updateUnderlyingResources(reqCtx intctrlutil.RequestCtx, cli client.Client, stsObj *appsv1.StatefulSet) error {
- if stsObj == nil {
- c.createWorkload()
- c.SetStatusPhase(appsv1alpha1.UpdatingClusterCompPhase, nil, "Component workload created")
- } else {
- if c.updateWorkload(stsObj) {
- c.SetStatusPhase(appsv1alpha1.UpdatingClusterCompPhase, nil, "Component workload updated")
- }
- // to work around that the scaled PVC will be deleted at object action.
- if err := c.updateVolumes(reqCtx, cli, stsObj); err != nil {
- return err
- }
- }
- if err := c.UpdatePDB(reqCtx, cli); err != nil {
- return err
- }
- if err := c.UpdateService(reqCtx, cli); err != nil {
- return err
- }
- // update KB___ env needed by pod to obtain hostname.
- c.updatePodEnvConfig()
- return nil
-}
-
-func (c *statefulComponentBase) createWorkload() {
- stsProto := c.WorkloadVertex.Obj.(*appsv1.StatefulSet)
- c.WorkloadVertex.Obj = stsProto
- c.WorkloadVertex.Action = ictrltypes.ActionCreatePtr()
-}
-
-func (c *statefulComponentBase) updateWorkload(stsObj *appsv1.StatefulSet) bool {
- stsObjCopy := stsObj.DeepCopy()
- stsProto := c.WorkloadVertex.Obj.(*appsv1.StatefulSet)
-
- // keep the original template annotations.
- // if annotations exist and are replaced, the statefulSet will be updated.
- mergeAnnotations(stsObjCopy.Spec.Template.Annotations, &stsProto.Spec.Template.Annotations)
- buildWorkLoadAnnotations(stsObjCopy, c.Cluster)
- stsObjCopy.Spec.Template = stsProto.Spec.Template
- stsObjCopy.Spec.Replicas = stsProto.Spec.Replicas
- c.updateUpdateStrategy(stsObjCopy, stsProto)
-
- resolvePodSpecDefaultFields(stsObj.Spec.Template.Spec, &stsObjCopy.Spec.Template.Spec)
-
- delayUpdatePodSpecSystemFields(stsObj.Spec.Template.Spec, &stsObjCopy.Spec.Template.Spec)
-
- if !reflect.DeepEqual(&stsObj.Spec, &stsObjCopy.Spec) {
- updatePodSpecSystemFields(&stsObjCopy.Spec.Template.Spec)
- c.WorkloadVertex.Obj = stsObjCopy
- c.WorkloadVertex.Action = ictrltypes.ActionPtr(ictrltypes.UPDATE)
- return true
- }
- return false
-}
-
-func (c *statefulComponentBase) updateUpdateStrategy(stsObj, stsProto *appsv1.StatefulSet) {
- var objMaxUnavailable *intstr.IntOrString
- if stsObj.Spec.UpdateStrategy.RollingUpdate != nil {
- objMaxUnavailable = stsObj.Spec.UpdateStrategy.RollingUpdate.MaxUnavailable
- }
- stsObj.Spec.UpdateStrategy = stsProto.Spec.UpdateStrategy
- if objMaxUnavailable == nil && stsObj.Spec.UpdateStrategy.RollingUpdate != nil {
- // HACK: This field is alpha-level (since v1.24) and is only honored by servers that enable the
- // MaxUnavailableStatefulSet feature.
- // When we get a nil MaxUnavailable from k8s, we consider that the field is not supported by the server,
- // and set the MaxUnavailable as nil explicitly to avoid the workload been updated unexpectedly.
- // Ref: https://kubernetes.io/docs/concepts/workloads/controllers/statefulset/#maximum-unavailable-pods
- stsObj.Spec.UpdateStrategy.RollingUpdate.MaxUnavailable = nil
- }
-}
-
-func (c *statefulComponentBase) updateVolumes(reqCtx intctrlutil.RequestCtx, cli client.Client, stsObj *appsv1.StatefulSet) error {
- // PVCs which have been added to the dag because of volume expansion.
- pvcNameSet := sets.New[string]()
- for _, v := range ictrltypes.FindAll[*corev1.PersistentVolumeClaim](c.Dag) {
- pvcNameSet.Insert(v.(*ictrltypes.LifecycleVertex).Obj.GetName())
- }
-
- for _, vct := range c.Component.VolumeClaimTemplates {
- pvcs, err := c.getRunningVolumes(reqCtx, cli, vct.Name, stsObj)
- if err != nil {
- return err
- }
- for _, pvc := range pvcs {
- if pvcNameSet.Has(pvc.Name) {
- continue
- }
- c.NoopResource(pvc, c.WorkloadVertex)
- }
- }
- return nil
-}
-
-func (c *statefulComponentBase) getRunningVolumes(reqCtx intctrlutil.RequestCtx, cli client.Client, vctName string,
- stsObj *appsv1.StatefulSet) ([]*corev1.PersistentVolumeClaim, error) {
- pvcs, err := listObjWithLabelsInNamespace(reqCtx.Ctx, cli, generics.PersistentVolumeClaimSignature, c.GetNamespace(), c.GetMatchingLabels())
- if err != nil {
- if apierrors.IsNotFound(err) {
- return nil, nil
- }
- return nil, err
- }
- matchedPVCs := make([]*corev1.PersistentVolumeClaim, 0)
- prefix := fmt.Sprintf("%s-%s", vctName, stsObj.Name)
- for _, pvc := range pvcs {
- if strings.HasPrefix(pvc.Name, prefix) {
- matchedPVCs = append(matchedPVCs, pvc)
- }
- }
- return matchedPVCs, nil
-}
diff --git a/controllers/apps/components/component.go b/controllers/apps/components/component.go
index 8ecd98a0e70..1469cab519b 100644
--- a/controllers/apps/components/component.go
+++ b/controllers/apps/components/component.go
@@ -22,116 +22,1498 @@ package components
import (
"context"
"fmt"
+ "reflect"
+ "strconv"
+ "strings"
"time"
+ "golang.org/x/exp/maps"
+ "golang.org/x/exp/slices"
+ appsv1 "k8s.io/api/apps/v1"
corev1 "k8s.io/api/core/v1"
+ policyv1 "k8s.io/api/policy/v1"
+ apierrors "k8s.io/apimachinery/pkg/api/errors"
metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
+ "k8s.io/apimachinery/pkg/runtime"
+ "k8s.io/apimachinery/pkg/runtime/schema"
+ "k8s.io/apimachinery/pkg/types"
+ "k8s.io/apimachinery/pkg/util/intstr"
+ "k8s.io/apimachinery/pkg/util/sets"
+ "k8s.io/client-go/tools/record"
"k8s.io/kubectl/pkg/util/podutils"
"sigs.k8s.io/controller-runtime/pkg/client"
+ "sigs.k8s.io/controller-runtime/pkg/client/apiutil"
appsv1alpha1 "github.com/apecloud/kubeblocks/apis/apps/v1alpha1"
- "github.com/apecloud/kubeblocks/internal/class"
+ dataprotectionv1alpha1 "github.com/apecloud/kubeblocks/apis/dataprotection/v1alpha1"
+ workloads "github.com/apecloud/kubeblocks/apis/workloads/v1alpha1"
+ cfgcore "github.com/apecloud/kubeblocks/internal/configuration/core"
+ "github.com/apecloud/kubeblocks/internal/configuration/util"
"github.com/apecloud/kubeblocks/internal/constant"
- types2 "github.com/apecloud/kubeblocks/internal/controller/client"
"github.com/apecloud/kubeblocks/internal/controller/component"
"github.com/apecloud/kubeblocks/internal/controller/graph"
- "github.com/apecloud/kubeblocks/internal/controller/plan"
+ rsmcore "github.com/apecloud/kubeblocks/internal/controller/rsm"
+ ictrltypes "github.com/apecloud/kubeblocks/internal/controller/types"
intctrlutil "github.com/apecloud/kubeblocks/internal/controllerutil"
+ "github.com/apecloud/kubeblocks/internal/generics"
+ viper "github.com/apecloud/kubeblocks/internal/viperx"
+ lorry "github.com/apecloud/kubeblocks/lorry/client"
)
-// PodIsAvailable checks whether a pod is available with respect to the workload type.
-// Deprecated: provide for ops request using, remove this interface later.
-func PodIsAvailable(workloadType appsv1alpha1.WorkloadType, pod *corev1.Pod, minReadySeconds int32) bool {
- if pod == nil {
- return false
+const (
+ // componentPhaseTransition the event reason indicates that the component transits to a new phase.
+ componentPhaseTransition = "ComponentPhaseTransition"
+
+ // podContainerFailedTimeout the timeout for container of pod failures, the component phase will be set to Failed/Abnormal after this time.
+ podContainerFailedTimeout = 10 * time.Second
+
+ // podScheduledFailedTimeout timeout for scheduling failure.
+ podScheduledFailedTimeout = 30 * time.Second
+)
+
+// rsmComponent as a base class for single rsm based component (stateful & replication & consensus).
+type rsmComponent struct {
+ Client client.Client
+ Recorder record.EventRecorder
+ Cluster *appsv1alpha1.Cluster
+ clusterVersion *appsv1alpha1.ClusterVersion // building config needs the cluster version
+ component *component.SynthesizedComponent // built synthesized component, replace it with component workload proto
+ dag *graph.DAG
+ workloadVertex *ictrltypes.LifecycleVertex // DAG vertex of main workload object
+ // runningWorkload can be nil, and the replicas of workload can be nil (zero)
+ runningWorkload *workloads.ReplicatedStateMachine
+}
+
+var _ Component = &rsmComponent{}
+
+func newRSMComponent(cli client.Client,
+ recorder record.EventRecorder,
+ cluster *appsv1alpha1.Cluster,
+ clusterVersion *appsv1alpha1.ClusterVersion,
+ synthesizedComponent *component.SynthesizedComponent,
+ dag *graph.DAG) Component {
+ comp := &rsmComponent{
+ Client: cli,
+ Recorder: recorder,
+ Cluster: cluster,
+ clusterVersion: clusterVersion,
+ component: synthesizedComponent,
+ dag: dag,
+ workloadVertex: nil,
+ }
+ return comp
+}
+
+func (c *rsmComponent) GetName() string {
+ return c.component.Name
+}
+
+func (c *rsmComponent) GetNamespace() string {
+ return c.Cluster.Namespace
+}
+
+func (c *rsmComponent) GetClusterName() string {
+ return c.Cluster.Name
+}
+
+func (c *rsmComponent) GetCluster() *appsv1alpha1.Cluster {
+ return c.Cluster
+}
+
+func (c *rsmComponent) GetClusterVersion() *appsv1alpha1.ClusterVersion {
+ return c.clusterVersion
+}
+
+func (c *rsmComponent) GetSynthesizedComponent() *component.SynthesizedComponent {
+ return c.component
+}
+
+func (c *rsmComponent) Create(reqCtx intctrlutil.RequestCtx, cli client.Client) error {
+ return c.create(reqCtx, cli, c.newBuilder(reqCtx, cli, ictrltypes.ActionCreatePtr()))
+}
+
+func (c *rsmComponent) Update(reqCtx intctrlutil.RequestCtx, cli client.Client) error {
+ return c.update(reqCtx, cli, c.newBuilder(reqCtx, cli, nil))
+}
+
+func (c *rsmComponent) Delete(reqCtx intctrlutil.RequestCtx, cli client.Client) error {
+ // TODO(impl): delete component owned resources
+ return nil
+}
+
+func (c *rsmComponent) Status(reqCtx intctrlutil.RequestCtx, cli client.Client) error {
+ return c.status(reqCtx, cli, c.newBuilder(reqCtx, cli, ictrltypes.ActionNoopPtr()))
+}
+
+func (c *rsmComponent) newBuilder(reqCtx intctrlutil.RequestCtx, cli client.Client,
+ action *ictrltypes.LifecycleAction) componentWorkloadBuilder {
+ return &rsmComponentWorkloadBuilder{
+ reqCtx: reqCtx,
+ client: cli,
+ comp: c,
+ defaultAction: action,
+ error: nil,
+ envConfig: nil,
+ workload: nil,
+ }
+}
+
+func (c *rsmComponent) setWorkload(obj client.Object, action *ictrltypes.LifecycleAction, parent *ictrltypes.LifecycleVertex) {
+ c.workloadVertex = c.addResource(obj, action, parent)
+}
+
+func (c *rsmComponent) addResource(obj client.Object, action *ictrltypes.LifecycleAction,
+ parent *ictrltypes.LifecycleVertex) *ictrltypes.LifecycleVertex {
+ if obj == nil {
+ panic("try to add nil object")
+ }
+ vertex := &ictrltypes.LifecycleVertex{
+ Obj: obj,
+ Action: action,
+ }
+ c.dag.AddVertex(vertex)
+
+ if parent != nil {
+ c.dag.Connect(parent, vertex)
+ }
+ return vertex
+}
+
+func (c *rsmComponent) init(reqCtx intctrlutil.RequestCtx, cli client.Client, builder componentWorkloadBuilder, load bool) error {
+ var err error
+ if builder != nil {
+ if err = builder.BuildEnv().
+ BuildWorkload().
+ BuildPDB().
+ BuildConfig().
+ BuildTLSVolume().
+ BuildVolumeMount().
+ BuildTLSCert().
+ Complete(); err != nil {
+ return err
+ }
+ }
+ if load {
+ c.runningWorkload, err = c.loadRunningWorkload(reqCtx, cli)
+ if err != nil {
+ return err
+ }
}
- switch workloadType {
- case appsv1alpha1.Consensus, appsv1alpha1.Replication:
- return intctrlutil.PodIsReadyWithLabel(*pod)
- case appsv1alpha1.Stateful, appsv1alpha1.Stateless:
- return podutils.IsPodAvailable(pod, minReadySeconds, metav1.Time{Time: time.Now()})
+ return nil
+}
+
+func (c *rsmComponent) loadRunningWorkload(reqCtx intctrlutil.RequestCtx, cli client.Client) (*workloads.ReplicatedStateMachine, error) {
+ rsmList, err := listRSMOwnedByComponent(reqCtx.Ctx, cli, c.GetNamespace(), c.getMatchingLabels())
+ if err != nil {
+ return nil, err
+ }
+ cnt := len(rsmList)
+ switch {
+ case cnt == 0:
+ return nil, nil
+ case cnt == 1:
+ return rsmList[0], nil
default:
- panic("unknown workload type")
+ return nil, fmt.Errorf("more than one workloads found for the component, cluster: %s, component: %s, cnt: %d",
+ c.GetClusterName(), c.GetName(), cnt)
}
}
-func NewComponent(reqCtx intctrlutil.RequestCtx,
- cli client.Client,
- definition *appsv1alpha1.ClusterDefinition,
- version *appsv1alpha1.ClusterVersion,
- cluster *appsv1alpha1.Cluster,
- compName string,
- dag *graph.DAG) (Component, error) {
- var compDef *appsv1alpha1.ClusterComponentDefinition
- var compVer *appsv1alpha1.ClusterComponentVersion
- compSpec := cluster.Spec.GetComponentByName(compName)
- if compSpec != nil {
- compDef = definition.GetComponentDefByName(compSpec.ComponentDefRef)
- if compDef == nil {
- return nil, fmt.Errorf("referenced component definition does not exist, cluster: %s, component: %s, component definition ref:%s",
- cluster.Name, compSpec.Name, compSpec.ComponentDefRef)
- }
- if version != nil {
- compVer = version.Spec.GetDefNameMappingComponents()[compSpec.ComponentDefRef]
+func (c *rsmComponent) getMatchingLabels() client.MatchingLabels {
+ return client.MatchingLabels{
+ constant.AppManagedByLabelKey: constant.AppName,
+ constant.AppInstanceLabelKey: c.GetClusterName(),
+ constant.KBAppComponentLabelKey: c.GetName(),
+ }
+}
+
+func (c *rsmComponent) create(reqCtx intctrlutil.RequestCtx, cli client.Client, builder componentWorkloadBuilder) error {
+ if err := c.init(reqCtx, cli, builder, false); err != nil {
+ return err
+ }
+
+ if err := c.validateObjectsAction(); err != nil {
+ return err
+ }
+
+ return nil
+}
+
+func (c *rsmComponent) update(reqCtx intctrlutil.RequestCtx, cli client.Client, builder componentWorkloadBuilder) error {
+ if err := c.init(reqCtx, cli, builder, true); err != nil {
+ return err
+ }
+
+ if c.runningWorkload != nil {
+ if err := c.restart(reqCtx, cli); err != nil {
+ return err
+ }
+
+ // cluster.spec.componentSpecs[*].volumeClaimTemplates[*].spec.resources.requests[corev1.ResourceStorage]
+ if err := c.expandVolume(reqCtx, cli); err != nil {
+ return err
+ }
+
+ // cluster.spec.componentSpecs[*].replicas
+ if err := c.horizontalScale(reqCtx, cli); err != nil {
+ return err
+ }
+ }
+
+ if err := c.updateUnderlyingResources(reqCtx, cli, c.runningWorkload); err != nil {
+ return err
+ }
+
+ return c.resolveObjectsAction(reqCtx, cli)
+}
+
+func (c *rsmComponent) status(reqCtx intctrlutil.RequestCtx, cli client.Client, builder componentWorkloadBuilder) error {
+ if err := c.init(reqCtx, cli, builder, true); err != nil {
+ return err
+ }
+ if c.runningWorkload == nil {
+ return nil
+ }
+
+ isDeleting := func() bool {
+ return !c.runningWorkload.DeletionTimestamp.IsZero()
+ }()
+ isZeroReplica := func() bool {
+ return (c.runningWorkload.Spec.Replicas == nil || *c.runningWorkload.Spec.Replicas == 0) && c.component.Replicas == 0
+ }()
+ pods, err := listPodOwnedByComponent(reqCtx.Ctx, cli, c.GetNamespace(), c.getMatchingLabels())
+ if err != nil {
+ return err
+ }
+ hasComponentPod := func() bool {
+ return len(pods) > 0
+ }()
+ isRunning, err := c.isRunning(reqCtx.Ctx, cli, c.runningWorkload)
+ if err != nil {
+ return err
+ }
+ isAllConfigSynced := c.isAllConfigSynced(reqCtx, cli)
+ hasFailedPod, messages, err := c.hasFailedPod(reqCtx, cli, pods)
+ if err != nil {
+ return err
+ }
+ isScaleOutFailed, err := c.isScaleOutFailed(reqCtx, cli)
+ if err != nil {
+ return err
+ }
+ hasRunningVolumeExpansion, hasFailedVolumeExpansion, err := c.hasVolumeExpansionRunning(reqCtx, cli)
+ if err != nil {
+ return err
+ }
+ hasFailure := func() bool {
+ return hasFailedPod || isScaleOutFailed || hasFailedVolumeExpansion
+ }()
+ isComponentAvailable, err := c.isAvailable(reqCtx, cli, pods)
+ if err != nil {
+ return err
+ }
+ isInCreatingPhase := func() bool {
+ phase := c.getComponentStatus().Phase
+ return phase == "" || phase == appsv1alpha1.CreatingClusterCompPhase
+ }()
+
+ updatePodsReady := func(ready bool) {
+ _ = c.updateStatus("", func(status *appsv1alpha1.ClusterComponentStatus) error {
+ // if ready flag not changed, don't update the ready time
+ if status.PodsReady != nil && *status.PodsReady == ready {
+ return nil
+ }
+ status.PodsReady = &ready
+ if ready {
+ now := metav1.Now()
+ status.PodsReadyTime = &now
+ }
+ return nil
+ })
+ }
+
+ podsReady := false
+ switch {
+ case isDeleting:
+ c.setStatusPhase(appsv1alpha1.DeletingClusterCompPhase, nil, "component is Deleting")
+ case isZeroReplica && hasComponentPod:
+ c.setStatusPhase(appsv1alpha1.StoppingClusterCompPhase, nil, "component is Stopping")
+ podsReady = true
+ case isZeroReplica:
+ c.setStatusPhase(appsv1alpha1.StoppedClusterCompPhase, nil, "component is Stopped")
+ podsReady = true
+ case isRunning && isAllConfigSynced && !hasRunningVolumeExpansion:
+ c.setStatusPhase(appsv1alpha1.RunningClusterCompPhase, nil, "component is Running")
+ podsReady = true
+ case !hasFailure && isInCreatingPhase:
+ c.setStatusPhase(appsv1alpha1.CreatingClusterCompPhase, nil, "Create a new component")
+ case !hasFailure:
+ c.setStatusPhase(appsv1alpha1.UpdatingClusterCompPhase, nil, "component is Updating")
+ case !isComponentAvailable:
+ c.setStatusPhase(appsv1alpha1.FailedClusterCompPhase, messages, "component is Failed")
+ default:
+ c.setStatusPhase(appsv1alpha1.AbnormalClusterCompPhase, nil, "unknown")
+ }
+ updatePodsReady(podsReady)
+
+ c.updateMembersStatus()
+
+ // works should continue to be done after spec updated.
+ if err := c.horizontalScale(reqCtx, cli); err != nil {
+ return err
+ }
+
+ c.updateWorkload(c.runningWorkload)
+
+ // update component info to pods' annotations
+ if err := updateComponentInfoToPods(reqCtx.Ctx, cli, c.Cluster, c.component, c.dag); err != nil {
+ return err
+ }
+
+ // patch the current componentSpec workload's custom labels
+ if err := updateCustomLabelToPods(reqCtx.Ctx, cli, c.Cluster, c.component, c.dag); err != nil {
+ reqCtx.Event(c.Cluster, corev1.EventTypeWarning, "component Workload Controller PatchWorkloadCustomLabelFailed", err.Error())
+ return err
+ }
+
+ return nil
+}
+
+func (c *rsmComponent) createResource(obj client.Object, parent *ictrltypes.LifecycleVertex) *ictrltypes.LifecycleVertex {
+ return ictrltypes.LifecycleObjectCreate(c.dag, obj, parent)
+}
+
+func (c *rsmComponent) deleteResource(obj client.Object, parent *ictrltypes.LifecycleVertex) *ictrltypes.LifecycleVertex {
+ return ictrltypes.LifecycleObjectDelete(c.dag, obj, parent)
+}
+
+func (c *rsmComponent) updateResource(obj client.Object, parent *ictrltypes.LifecycleVertex) *ictrltypes.LifecycleVertex {
+ return ictrltypes.LifecycleObjectUpdate(c.dag, obj, parent)
+}
+
+func (c *rsmComponent) patchResource(obj client.Object, objCopy client.Object, parent *ictrltypes.LifecycleVertex) *ictrltypes.LifecycleVertex {
+ return ictrltypes.LifecycleObjectPatch(c.dag, obj, objCopy, parent)
+}
+
+func (c *rsmComponent) noopResource(obj client.Object, parent *ictrltypes.LifecycleVertex) *ictrltypes.LifecycleVertex {
+ return ictrltypes.LifecycleObjectNoop(c.dag, obj, parent)
+}
+
+// validateObjectsAction validates the action of objects in dag has been determined.
+func (c *rsmComponent) validateObjectsAction() error {
+ for _, v := range c.dag.Vertices() {
+ node, ok := v.(*ictrltypes.LifecycleVertex)
+ if !ok {
+ return fmt.Errorf("unexpected vertex type, cluster: %s, component: %s, vertex: %T",
+ c.GetClusterName(), c.GetName(), v)
+ }
+ if node.Obj == nil {
+ return fmt.Errorf("unexpected nil vertex object, cluster: %s, component: %s, vertex: %T",
+ c.GetClusterName(), c.GetName(), v)
+ }
+ if node.Action == nil {
+ return fmt.Errorf("unexpected nil vertex action, cluster: %s, component: %s, vertex: %T",
+ c.GetClusterName(), c.GetName(), v)
+ }
+ }
+ return nil
+}
+
+// resolveObjectsAction resolves the action of objects in dag to guarantee that all object actions will be determined.
+func (c *rsmComponent) resolveObjectsAction(reqCtx intctrlutil.RequestCtx, cli client.Client) error {
+ snapshot, err := readCacheSnapshot(reqCtx, cli, c.GetCluster())
+ if err != nil {
+ return err
+ }
+ for _, v := range c.dag.Vertices() {
+ node, ok := v.(*ictrltypes.LifecycleVertex)
+ if !ok {
+ return fmt.Errorf("unexpected vertex type, cluster: %s, component: %s, vertex: %T",
+ c.GetClusterName(), c.GetName(), v)
+ }
+ if node.Action == nil {
+ if action, err := resolveObjectAction(snapshot, node, cli.Scheme()); err != nil {
+ return err
+ } else {
+ node.Action = action
+ }
+ }
+ }
+ if c.GetCluster().IsStatusUpdating() {
+ for _, vertex := range c.dag.Vertices() {
+ v, _ := vertex.(*ictrltypes.LifecycleVertex)
+ // TODO(refactor): fix me, this is a workaround for h-scaling to update stateful set.
+ if _, ok := v.Obj.(*appsv1.StatefulSet); !ok {
+ v.Immutable = true
+ }
+ }
+ }
+ return c.validateObjectsAction()
+}
+
+// setStatusPhase sets the cluster component phase and messages conditionally.
+func (c *rsmComponent) setStatusPhase(phase appsv1alpha1.ClusterComponentPhase,
+ statusMessage appsv1alpha1.ComponentMessageMap, phaseTransitionMsg string) {
+ updatefn := func(status *appsv1alpha1.ClusterComponentStatus) error {
+ if status.Phase == phase {
+ return nil
+ }
+ status.Phase = phase
+ if status.Message == nil {
+ status.Message = statusMessage
+ } else {
+ for k, v := range statusMessage {
+ status.Message[k] = v
+ }
+ }
+ return nil
+ }
+ if err := c.updateStatus(phaseTransitionMsg, updatefn); err != nil {
+ panic(fmt.Sprintf("unexpected error occurred while updating component status: %s", err.Error()))
+ }
+}
+
+// updateStatus updates the cluster component status by @updatefn, with additional message to explain the transition occurred.
+func (c *rsmComponent) updateStatus(phaseTransitionMsg string, updatefn func(status *appsv1alpha1.ClusterComponentStatus) error) error {
+ if updatefn == nil {
+ return nil
+ }
+
+ status := c.getComponentStatus()
+ phase := status.Phase
+ err := updatefn(&status)
+ if err != nil {
+ return err
+ }
+ c.Cluster.Status.Components[c.GetName()] = status
+
+ if phase != status.Phase {
+ // TODO: logging the event
+ if c.Recorder != nil && phaseTransitionMsg != "" {
+ c.Recorder.Eventf(c.Cluster, corev1.EventTypeNormal, componentPhaseTransition, phaseTransitionMsg)
+ }
+ }
+
+ return nil
+}
+
+func (c *rsmComponent) isRunning(ctx context.Context, cli client.Client, obj client.Object) (bool, error) {
+ if obj == nil {
+ return false, nil
+ }
+ rsm, ok := obj.(*workloads.ReplicatedStateMachine)
+ if !ok {
+ return false, nil
+ }
+ if isLatestRevision, err := IsComponentPodsWithLatestRevision(ctx, cli, c.Cluster, rsm); err != nil {
+ return false, err
+ } else if !isLatestRevision {
+ return false, nil
+ }
+
+ // whether rsm is ready
+ return rsmcore.IsRSMReady(rsm), nil
+}
+
+// isAvailable tells whether the component is basically available, ether working well or in a fragile state:
+// 1. at least one pod is available
+// 2. with latest revision
+// 3. and with leader role label set
+func (c *rsmComponent) isAvailable(reqCtx intctrlutil.RequestCtx, cli client.Client, pods []*corev1.Pod) (bool, error) {
+ if isLatestRevision, err := IsComponentPodsWithLatestRevision(reqCtx.Ctx, cli, c.Cluster, c.runningWorkload); err != nil {
+ return false, err
+ } else if !isLatestRevision {
+ return false, nil
+ }
+
+ shouldCheckLeader := func() bool {
+ return c.component.WorkloadType == appsv1alpha1.Consensus || c.component.WorkloadType == appsv1alpha1.Replication
+ }()
+ hasLeaderRoleLabel := func(pod *corev1.Pod) bool {
+ roleName, ok := pod.Labels[constant.RoleLabelKey]
+ if !ok {
+ return false
+ }
+ for _, replicaRole := range c.runningWorkload.Spec.Roles {
+ if roleName == replicaRole.Name && replicaRole.IsLeader {
+ return true
+ }
+ }
+ return false
+ }
+ for _, pod := range pods {
+ if !podutils.IsPodAvailable(pod, 0, metav1.Time{Time: time.Now()}) {
+ continue
+ }
+ if !shouldCheckLeader {
+ continue
+ }
+ if _, ok := pod.Labels[constant.RoleLabelKey]; ok {
+ continue
+ }
+ if hasLeaderRoleLabel(pod) {
+ return true, nil
+ }
+ }
+ return false, nil
+}
+
+func (c *rsmComponent) hasFailedPod(reqCtx intctrlutil.RequestCtx, cli client.Client, pods []*corev1.Pod) (bool, appsv1alpha1.ComponentMessageMap, error) {
+ if isLatestRevision, err := IsComponentPodsWithLatestRevision(reqCtx.Ctx, cli, c.Cluster, c.runningWorkload); err != nil {
+ return false, nil, err
+ } else if !isLatestRevision {
+ return false, nil, nil
+ }
+
+ var messages appsv1alpha1.ComponentMessageMap
+ // check pod readiness
+ hasFailedPod, msg, _ := hasFailedAndTimedOutPod(pods)
+ if hasFailedPod {
+ messages = msg
+ return true, messages, nil
+ }
+ // check role probe
+ if c.component.WorkloadType != appsv1alpha1.Consensus && c.component.WorkloadType != appsv1alpha1.Replication {
+ return false, messages, nil
+ }
+ hasProbeTimeout := false
+ for _, pod := range pods {
+ if _, ok := pod.Labels[constant.RoleLabelKey]; ok {
+ continue
+ }
+ for _, condition := range pod.Status.Conditions {
+ if condition.Type != corev1.PodReady || condition.Status != corev1.ConditionTrue {
+ continue
+ }
+ podsReadyTime := &condition.LastTransitionTime
+ if isProbeTimeout(c.component.Probes, podsReadyTime) {
+ hasProbeTimeout = true
+ if messages == nil {
+ messages = appsv1alpha1.ComponentMessageMap{}
+ }
+ messages.SetObjectMessage(pod.Kind, pod.Name, "Role probe timeout, check whether the application is available")
+ }
+ }
+ }
+ return hasProbeTimeout, messages, nil
+}
+
+func (c *rsmComponent) isAllConfigSynced(reqCtx intctrlutil.RequestCtx, cli client.Client) bool {
+ checkFinishedReconfigure := func(cm *corev1.ConfigMap) bool {
+ labels := cm.GetLabels()
+ annotations := cm.GetAnnotations()
+ if len(annotations) == 0 || len(labels) == 0 {
+ return false
+ }
+ hash, _ := util.ComputeHash(cm.Data)
+ return labels[constant.CMInsConfigurationHashLabelKey] == hash
+ }
+
+ var (
+ cmKey client.ObjectKey
+ cmObj = &corev1.ConfigMap{}
+ allConfigSynced = true
+ )
+ for _, configSpec := range c.component.ConfigTemplates {
+ cmKey = client.ObjectKey{
+ Namespace: c.GetNamespace(),
+ Name: cfgcore.GetComponentCfgName(c.GetClusterName(), c.GetName(), configSpec.Name),
+ }
+ if err := cli.Get(reqCtx.Ctx, cmKey, cmObj); err != nil {
+ return true
+ }
+ if !checkFinishedReconfigure(cmObj) {
+ allConfigSynced = false
+ break
+ }
+ }
+ return allConfigSynced
+}
+
+func (c *rsmComponent) updateMembersStatus() {
+ // get component status
+ componentStatus := c.getComponentStatus()
+
+ // for compatibilities prior KB 0.7.0
+ buildConsensusSetStatus := func(membersStatus []workloads.MemberStatus) *appsv1alpha1.ConsensusSetStatus {
+ consensusSetStatus := &appsv1alpha1.ConsensusSetStatus{
+ Leader: appsv1alpha1.ConsensusMemberStatus{
+ Name: "",
+ Pod: constant.ComponentStatusDefaultPodName,
+ AccessMode: appsv1alpha1.None,
+ },
+ }
+ for _, memberStatus := range membersStatus {
+ status := appsv1alpha1.ConsensusMemberStatus{
+ Name: memberStatus.Name,
+ Pod: memberStatus.PodName,
+ AccessMode: appsv1alpha1.AccessMode(memberStatus.AccessMode),
+ }
+ switch {
+ case memberStatus.IsLeader:
+ consensusSetStatus.Leader = status
+ case memberStatus.CanVote:
+ consensusSetStatus.Followers = append(consensusSetStatus.Followers, status)
+ default:
+ consensusSetStatus.Learner = &status
+ }
+ }
+ return consensusSetStatus
+ }
+ buildReplicationSetStatus := func(membersStatus []workloads.MemberStatus) *appsv1alpha1.ReplicationSetStatus {
+ replicationSetStatus := &appsv1alpha1.ReplicationSetStatus{
+ Primary: appsv1alpha1.ReplicationMemberStatus{
+ Pod: "Unknown",
+ },
+ }
+ for _, memberStatus := range membersStatus {
+ status := appsv1alpha1.ReplicationMemberStatus{
+ Pod: memberStatus.PodName,
+ }
+ switch {
+ case memberStatus.IsLeader:
+ replicationSetStatus.Primary = status
+ default:
+ replicationSetStatus.Secondaries = append(replicationSetStatus.Secondaries, status)
+ }
+ }
+ return replicationSetStatus
+ }
+
+ // update members status
+ switch c.component.WorkloadType {
+ case appsv1alpha1.Consensus:
+ componentStatus.ConsensusSetStatus = buildConsensusSetStatus(c.runningWorkload.Status.MembersStatus)
+ case appsv1alpha1.Replication:
+ componentStatus.ReplicationSetStatus = buildReplicationSetStatus(c.runningWorkload.Status.MembersStatus)
+ }
+ componentStatus.MembersStatus = slices.Clone(c.runningWorkload.Status.MembersStatus)
+
+ // set component status back
+ c.Cluster.Status.Components[c.GetName()] = componentStatus
+}
+
+func (c *rsmComponent) getComponentStatus() appsv1alpha1.ClusterComponentStatus {
+ if c.Cluster.Status.Components == nil {
+ c.Cluster.Status.Components = make(map[string]appsv1alpha1.ClusterComponentStatus)
+ }
+ if _, ok := c.Cluster.Status.Components[c.GetName()]; !ok {
+ c.Cluster.Status.Components[c.GetName()] = appsv1alpha1.ClusterComponentStatus{}
+ }
+ return c.Cluster.Status.Components[c.GetName()]
+}
+
+func (c *rsmComponent) isScaleOutFailed(reqCtx intctrlutil.RequestCtx, cli client.Client) (bool, error) {
+ if c.runningWorkload.Spec.Replicas == nil {
+ return false, nil
+ }
+ if c.component.Replicas <= *c.runningWorkload.Spec.Replicas {
+ return false, nil
+ }
+ if c.workloadVertex == nil {
+ return false, nil
+ }
+ stsObj := ConvertRSMToSTS(c.runningWorkload)
+ rsmProto := c.workloadVertex.Obj.(*workloads.ReplicatedStateMachine)
+ stsProto := ConvertRSMToSTS(rsmProto)
+ backupKey := types.NamespacedName{
+ Namespace: stsObj.Namespace,
+ Name: stsObj.Name + "-scaling",
+ }
+ d, err := newDataClone(reqCtx, cli, c.Cluster, c.component, stsObj, stsProto, backupKey)
+ if err != nil {
+ return false, err
+ }
+ if status, err := d.checkBackupStatus(); err != nil {
+ return false, err
+ } else if status == backupStatusFailed {
+ return true, nil
+ }
+ for i := *c.runningWorkload.Spec.Replicas; i < c.component.Replicas; i++ {
+ if status, err := d.checkRestoreStatus(i); err != nil {
+ return false, err
+ } else if status == backupStatusFailed {
+ return true, nil
+ }
+ }
+ return false, nil
+}
+
+func (c *rsmComponent) restart(reqCtx intctrlutil.RequestCtx, cli client.Client) error {
+ return restartPod(&c.runningWorkload.Spec.Template)
+}
+
+func (c *rsmComponent) expandVolume(reqCtx intctrlutil.RequestCtx, cli client.Client) error {
+ for _, vct := range c.runningWorkload.Spec.VolumeClaimTemplates {
+ var proto *corev1.PersistentVolumeClaimTemplate
+ for _, v := range c.component.VolumeClaimTemplates {
+ if v.Name == vct.Name {
+ proto = &v
+ break
+ }
+ }
+ // REVIEW: seems we can remove a volume claim from templates at runtime, without any changes and warning messages?
+ if proto == nil {
+ continue
+ }
+
+ if err := c.expandVolumes(reqCtx, cli, vct.Name, proto); err != nil {
+ return err
+ }
+ }
+ return nil
+}
+
+func (c *rsmComponent) expandVolumes(reqCtx intctrlutil.RequestCtx, cli client.Client,
+ vctName string, proto *corev1.PersistentVolumeClaimTemplate) error {
+ pvcNotFound := false
+ for i := *c.runningWorkload.Spec.Replicas - 1; i >= 0; i-- {
+ pvc := &corev1.PersistentVolumeClaim{}
+ pvcKey := types.NamespacedName{
+ Namespace: c.GetNamespace(),
+ Name: fmt.Sprintf("%s-%s-%d", vctName, c.runningWorkload.Name, i),
+ }
+ if err := cli.Get(reqCtx.Ctx, pvcKey, pvc); err != nil {
+ if apierrors.IsNotFound(err) {
+ pvcNotFound = true
+ } else {
+ return err
+ }
+ }
+ if err := c.updatePVCSize(reqCtx, cli, pvcKey, pvc, pvcNotFound, proto); err != nil {
+ return err
+ }
+ }
+ return nil
+}
+
+func (c *rsmComponent) updatePVCSize(reqCtx intctrlutil.RequestCtx, cli client.Client, pvcKey types.NamespacedName,
+ pvc *corev1.PersistentVolumeClaim, pvcNotFound bool, vctProto *corev1.PersistentVolumeClaimTemplate) error {
+ // reference: https://kubernetes.io/docs/concepts/storage/persistent-volumes/#recovering-from-failure-when-expanding-volumes
+ // 1. Mark the PersistentVolume(PV) that is bound to the PersistentVolumeClaim(PVC) with Retain reclaim policy.
+ // 2. Delete the PVC. Since PV has Retain reclaim policy - we will not lose any data when we recreate the PVC.
+ // 3. Delete the claimRef entry from PV specs, so as new PVC can bind to it. This should make the PV Available.
+ // 4. Re-create the PVC with smaller size than PV and set volumeName field of the PVC to the name of the PV. This should bind new PVC to existing PV.
+ // 5. Don't forget to restore the reclaim policy of the PV.
+ newPVC := pvc.DeepCopy()
+ if pvcNotFound {
+ newPVC.Name = pvcKey.Name
+ newPVC.Namespace = pvcKey.Namespace
+ newPVC.SetLabels(vctProto.Labels)
+ newPVC.Spec = vctProto.Spec
+ ml := client.MatchingLabels{
+ constant.PVCNameLabelKey: pvcKey.Name,
+ }
+ pvList := corev1.PersistentVolumeList{}
+ if err := cli.List(reqCtx.Ctx, &pvList, ml); err != nil {
+ return err
+ }
+ for _, pv := range pvList.Items {
+ // find pv referenced this pvc
+ if pv.Spec.ClaimRef == nil {
+ continue
+ }
+ if pv.Spec.ClaimRef.Name == pvcKey.Name {
+ newPVC.Spec.VolumeName = pv.Name
+ break
+ }
}
} else {
- compDef = definition.GetComponentDefByName(compName)
- if version != nil {
- compVer = version.Spec.GetDefNameMappingComponents()[compName]
+ newPVC.Spec.Resources.Requests[corev1.ResourceStorage] = vctProto.Spec.Resources.Requests[corev1.ResourceStorage]
+ // delete annotation to make it re-bind
+ delete(newPVC.Annotations, "pv.kubernetes.io/bind-completed")
+ }
+
+ pvNotFound := false
+
+ // step 1: update pv to retain
+ pv := &corev1.PersistentVolume{}
+ pvKey := types.NamespacedName{
+ Namespace: pvcKey.Namespace,
+ Name: newPVC.Spec.VolumeName,
+ }
+ if err := cli.Get(reqCtx.Ctx, pvKey, pv); err != nil {
+ if apierrors.IsNotFound(err) {
+ pvNotFound = true
+ } else {
+ return err
}
}
- if compDef == nil {
- return nil, nil
+ type pvcRecreateStep int
+ const (
+ pvPolicyRetainStep pvcRecreateStep = iota
+ deletePVCStep
+ removePVClaimRefStep
+ createPVCStep
+ pvRestorePolicyStep
+ )
+
+ addStepMap := map[pvcRecreateStep]func(fromVertex *ictrltypes.LifecycleVertex, step pvcRecreateStep) *ictrltypes.LifecycleVertex{
+ pvPolicyRetainStep: func(fromVertex *ictrltypes.LifecycleVertex, step pvcRecreateStep) *ictrltypes.LifecycleVertex {
+ // step 1: update pv to retain
+ retainPV := pv.DeepCopy()
+ if retainPV.Labels == nil {
+ retainPV.Labels = make(map[string]string)
+ }
+ // add label to pv, in case pvc get deleted, and we can't find pv
+ retainPV.Labels[constant.PVCNameLabelKey] = pvcKey.Name
+ if retainPV.Annotations == nil {
+ retainPV.Annotations = make(map[string]string)
+ }
+ retainPV.Annotations[constant.PVLastClaimPolicyAnnotationKey] = string(pv.Spec.PersistentVolumeReclaimPolicy)
+ retainPV.Spec.PersistentVolumeReclaimPolicy = corev1.PersistentVolumeReclaimRetain
+ return c.patchResource(retainPV, pv, fromVertex)
+ },
+ deletePVCStep: func(fromVertex *ictrltypes.LifecycleVertex, step pvcRecreateStep) *ictrltypes.LifecycleVertex {
+ // step 2: delete pvc, this will not delete pv because policy is 'retain'
+ removeFinalizerPVC := pvc.DeepCopy()
+ removeFinalizerPVC.SetFinalizers([]string{})
+ removeFinalizerPVCVertex := c.patchResource(removeFinalizerPVC, pvc, fromVertex)
+ return c.deleteResource(pvc, removeFinalizerPVCVertex)
+ },
+ removePVClaimRefStep: func(fromVertex *ictrltypes.LifecycleVertex, step pvcRecreateStep) *ictrltypes.LifecycleVertex {
+ // step 3: remove claimRef in pv
+ removeClaimRefPV := pv.DeepCopy()
+ if removeClaimRefPV.Spec.ClaimRef != nil {
+ removeClaimRefPV.Spec.ClaimRef.UID = ""
+ removeClaimRefPV.Spec.ClaimRef.ResourceVersion = ""
+ }
+ return c.patchResource(removeClaimRefPV, pv, fromVertex)
+ },
+ createPVCStep: func(fromVertex *ictrltypes.LifecycleVertex, step pvcRecreateStep) *ictrltypes.LifecycleVertex {
+ // step 4: create new pvc
+ newPVC.SetResourceVersion("")
+ return c.createResource(newPVC, fromVertex)
+ },
+ pvRestorePolicyStep: func(fromVertex *ictrltypes.LifecycleVertex, step pvcRecreateStep) *ictrltypes.LifecycleVertex {
+ // step 5: restore to previous pv policy
+ restorePV := pv.DeepCopy()
+ policy := corev1.PersistentVolumeReclaimPolicy(restorePV.Annotations[constant.PVLastClaimPolicyAnnotationKey])
+ if len(policy) == 0 {
+ policy = corev1.PersistentVolumeReclaimDelete
+ }
+ restorePV.Spec.PersistentVolumeReclaimPolicy = policy
+ return c.patchResource(restorePV, pv, fromVertex)
+ },
+ }
+
+ updatePVCByRecreateFromStep := func(fromStep pvcRecreateStep) {
+ lastVertex := c.workloadVertex
+ for step := pvRestorePolicyStep; step >= fromStep && step >= pvPolicyRetainStep; step-- {
+ lastVertex = addStepMap[step](lastVertex, step)
+ }
+ }
+
+ targetQuantity := vctProto.Spec.Resources.Requests[corev1.ResourceStorage]
+ if pvcNotFound && !pvNotFound {
+ // this could happen if create pvc step failed when recreating pvc
+ updatePVCByRecreateFromStep(removePVClaimRefStep)
+ return nil
+ }
+ if pvcNotFound && pvNotFound {
+ // if both pvc and pv not found, do nothing
+ return nil
+ }
+ if reflect.DeepEqual(pvc.Spec.Resources, newPVC.Spec.Resources) && pv.Spec.PersistentVolumeReclaimPolicy == corev1.PersistentVolumeReclaimRetain {
+ // this could happen if create pvc succeeded but last step failed
+ updatePVCByRecreateFromStep(pvRestorePolicyStep)
+ return nil
+ }
+ if pvcQuantity := pvc.Spec.Resources.Requests[corev1.ResourceStorage]; !viper.GetBool(constant.CfgRecoverVolumeExpansionFailure) &&
+ pvcQuantity.Cmp(targetQuantity) == 1 && // check if it's compressing volume
+ targetQuantity.Cmp(*pvc.Status.Capacity.Storage()) >= 0 { // check if target size is greater than or equal to actual size
+ // this branch means we can update pvc size by recreate it
+ updatePVCByRecreateFromStep(pvPolicyRetainStep)
+ return nil
+ }
+ if pvcQuantity := pvc.Spec.Resources.Requests[corev1.ResourceStorage]; pvcQuantity.Cmp(vctProto.Spec.Resources.Requests[corev1.ResourceStorage]) != 0 {
+ // use pvc's update without anything extra
+ c.updateResource(newPVC, c.workloadVertex)
+ return nil
+ }
+ // all the else means no need to update
+
+ return nil
+}
+
+func (c *rsmComponent) hasVolumeExpansionRunning(reqCtx intctrlutil.RequestCtx, cli client.Client) (bool, bool, error) {
+ var (
+ running bool
+ failed bool
+ )
+ for _, vct := range c.runningWorkload.Spec.VolumeClaimTemplates {
+ volumes, err := c.getRunningVolumes(reqCtx, cli, vct.Name, c.runningWorkload)
+ if err != nil {
+ return false, false, err
+ }
+ for _, v := range volumes {
+ if v.Status.Capacity == nil || v.Status.Capacity.Storage().Cmp(v.Spec.Resources.Requests[corev1.ResourceStorage]) >= 0 {
+ continue
+ }
+ running = true
+ // TODO: how to check the expansion failed?
+ }
}
+ return running, failed, nil
+}
- clsMgr, err := getClassManager(reqCtx.Ctx, cli, cluster)
+func (c *rsmComponent) horizontalScale(reqCtx intctrlutil.RequestCtx, cli client.Client) error {
+ sts := ConvertRSMToSTS(c.runningWorkload)
+ if sts.Status.ReadyReplicas == c.component.Replicas {
+ return nil
+ }
+ ret := c.horizontalScaling(sts)
+ if ret == 0 {
+ if err := c.postScaleIn(reqCtx, cli); err != nil {
+ return err
+ }
+ if err := c.postScaleOut(reqCtx, cli, sts); err != nil {
+ return err
+ }
+ return nil
+ }
+ if ret < 0 {
+ if err := c.scaleIn(reqCtx, cli, sts); err != nil {
+ return err
+ }
+ } else {
+ if err := c.scaleOut(reqCtx, cli, sts); err != nil {
+ return err
+ }
+ }
+
+ if err := c.updatePodReplicaLabel4Scaling(reqCtx, cli, c.component.Replicas); err != nil {
+ return err
+ }
+
+ // update KB___ env needed by pod to obtain hostname.
+ c.updatePodEnvConfig()
+
+ reqCtx.Recorder.Eventf(c.Cluster,
+ corev1.EventTypeNormal,
+ "HorizontalScale",
+ "start horizontal scale component %s of cluster %s from %d to %d",
+ c.GetName(), c.GetClusterName(), int(c.component.Replicas)-ret, c.component.Replicas)
+
+ return nil
+}
+
+// < 0 for scale in, > 0 for scale out, and == 0 for nothing
+func (c *rsmComponent) horizontalScaling(stsObj *appsv1.StatefulSet) int {
+ return int(c.component.Replicas - *stsObj.Spec.Replicas)
+}
+
+func (c *rsmComponent) updatePodEnvConfig() {
+ for _, v := range ictrltypes.FindAll[*corev1.ConfigMap](c.dag) {
+ node := v.(*ictrltypes.LifecycleVertex)
+ // TODO: need a way to reference the env config.
+ envConfigName := fmt.Sprintf("%s-%s-env", c.GetClusterName(), c.GetName())
+ if node.Obj.GetName() == envConfigName {
+ node.Action = ictrltypes.ActionUpdatePtr()
+ }
+ }
+}
+
+func (c *rsmComponent) updatePodReplicaLabel4Scaling(reqCtx intctrlutil.RequestCtx, cli client.Client, replicas int32) error {
+ pods, err := listPodOwnedByComponent(reqCtx.Ctx, cli, c.GetNamespace(), c.getMatchingLabels())
if err != nil {
- return nil, err
+ return err
+ }
+ for _, pod := range pods {
+ obj := pod.DeepCopy()
+ if obj.Annotations == nil {
+ obj.Annotations = make(map[string]string)
+ }
+ obj.Annotations[constant.ComponentReplicasAnnotationKey] = strconv.Itoa(int(replicas))
+ c.updateResource(obj, c.workloadVertex)
+ }
+ return nil
+}
+
+func (c *rsmComponent) scaleIn(reqCtx intctrlutil.RequestCtx, cli client.Client, stsObj *appsv1.StatefulSet) error {
+ // if scale in to 0, do not delete pvcs
+ if c.component.Replicas == 0 {
+ reqCtx.Log.Info("scale in to 0, keep all PVCs")
+ return nil
}
- serviceReferences, err := plan.GenServiceReferences(reqCtx, cli, cluster, compDef, compSpec)
+ // TODO: check the component definition to determine whether we need to call leave member before deleting replicas.
+ err := c.leaveMember4ScaleIn(reqCtx, cli, stsObj)
if err != nil {
- return nil, err
+ reqCtx.Log.Info(fmt.Sprintf("leave member at scaling-in error, retry later: %s", err.Error()))
+ return err
+ }
+ return c.deletePVCs4ScaleIn(reqCtx, cli, stsObj)
+}
+
+func (c *rsmComponent) postScaleIn(reqCtx intctrlutil.RequestCtx, cli client.Client) error {
+ return nil
+}
+
+func (c *rsmComponent) leaveMember4ScaleIn(reqCtx intctrlutil.RequestCtx, cli client.Client, stsObj *appsv1.StatefulSet) error {
+ pods, err := listPodOwnedByComponent(reqCtx.Ctx, cli, c.GetNamespace(), c.getMatchingLabels())
+ if err != nil {
+ return err
+ }
+ for _, pod := range pods {
+ subs := strings.Split(pod.Name, "-")
+ if ordinal, err := strconv.ParseInt(subs[len(subs)-1], 10, 32); err != nil {
+ return err
+ } else if int32(ordinal) < c.component.Replicas {
+ continue
+ }
+ lorryCli, err1 := lorry.NewClient(c.component.CharacterType, *pod)
+ if err1 != nil {
+ if err == nil {
+ err = err1
+ }
+ continue
+ }
+
+ if intctrlutil.IsNil(lorryCli) {
+ // no lorry in the pod
+ continue
+ }
+
+ if err2 := lorryCli.LeaveMember(reqCtx.Ctx); err2 != nil {
+ if err == nil {
+ err = err2
+ }
+ }
+ }
+ return err // TODO: use requeue-after
+}
+
+func (c *rsmComponent) deletePVCs4ScaleIn(reqCtx intctrlutil.RequestCtx, cli client.Client, stsObj *appsv1.StatefulSet) error {
+ for i := c.component.Replicas; i < *stsObj.Spec.Replicas; i++ {
+ for _, vct := range stsObj.Spec.VolumeClaimTemplates {
+ pvcKey := types.NamespacedName{
+ Namespace: stsObj.Namespace,
+ Name: fmt.Sprintf("%s-%s-%d", vct.Name, stsObj.Name, i),
+ }
+ pvc := corev1.PersistentVolumeClaim{}
+ if err := cli.Get(reqCtx.Ctx, pvcKey, &pvc); err != nil {
+ return err
+ }
+ // Since there are no order guarantee between updating STS and deleting PVCs, if there is any error occurred
+ // after updating STS and before deleting PVCs, the PVCs intended to scale-in will be leaked.
+ // For simplicity, the updating dependency is added between them to guarantee that the PVCs to scale-in
+ // will be deleted or the scaling-in operation will be failed.
+ c.deleteResource(&pvc, c.workloadVertex)
+ }
+ }
+ return nil
+}
+
+func (c *rsmComponent) scaleOut(reqCtx intctrlutil.RequestCtx, cli client.Client, stsObj *appsv1.StatefulSet) error {
+ var (
+ backupKey = types.NamespacedName{
+ Namespace: stsObj.Namespace,
+ Name: stsObj.Name + "-scaling",
+ }
+ )
+
+ // sts's replicas=0 means it's starting not scaling, skip all the scaling work.
+ if *stsObj.Spec.Replicas == 0 {
+ return nil
+ }
+
+ c.workloadVertex.Immutable = true
+ rsmProto := c.workloadVertex.Obj.(*workloads.ReplicatedStateMachine)
+ stsProto := ConvertRSMToSTS(rsmProto)
+ d, err := newDataClone(reqCtx, cli, c.Cluster, c.component, stsObj, stsProto, backupKey)
+ if err != nil {
+ return err
+ }
+ var succeed bool
+ if d == nil {
+ succeed = true
+ } else {
+ succeed, err = d.succeed()
+ if err != nil {
+ return err
+ }
+ }
+ if succeed {
+ // pvcs are ready, rsm.replicas should be updated
+ c.workloadVertex.Immutable = false
+ return c.postScaleOut(reqCtx, cli, stsObj)
+ } else {
+ c.workloadVertex.Immutable = true
+ // update objs will trigger cluster reconcile, no need to requeue error
+ objs, err := d.cloneData(d)
+ if err != nil {
+ return err
+ }
+ for _, obj := range objs {
+ c.createResource(obj, nil)
+ }
+ return nil
+ }
+}
+
+func (c *rsmComponent) postScaleOut(reqCtx intctrlutil.RequestCtx, cli client.Client, stsObj *appsv1.StatefulSet) error {
+ var (
+ snapshotKey = types.NamespacedName{
+ Namespace: stsObj.Namespace,
+ Name: stsObj.Name + "-scaling",
+ }
+ )
+
+ d, err := newDataClone(reqCtx, cli, c.Cluster, c.component, stsObj, stsObj, snapshotKey)
+ if err != nil {
+ return err
+ }
+ if d != nil {
+ // clean backup resources.
+ // there will not be any backup resources other than scale out.
+ tmpObjs, err := d.clearTmpResources()
+ if err != nil {
+ return err
+ }
+ for _, obj := range tmpObjs {
+ c.deleteResource(obj, nil)
+ }
+ }
+
+ return nil
+}
+
+func (c *rsmComponent) updateUnderlyingResources(reqCtx intctrlutil.RequestCtx, cli client.Client, rsmObj *workloads.ReplicatedStateMachine) error {
+ if rsmObj == nil {
+ c.createWorkload()
+ } else {
+ c.updateWorkload(rsmObj)
+ // to work around that the scaled PVC will be deleted at object action.
+ if err := c.updateVolumes(reqCtx, cli, rsmObj); err != nil {
+ return err
+ }
+ }
+ if err := c.updatePDB(reqCtx, cli); err != nil {
+ return err
+ }
+ return nil
+}
+
+func (c *rsmComponent) createWorkload() {
+ rsmProto := c.workloadVertex.Obj.(*workloads.ReplicatedStateMachine)
+ buildWorkLoadAnnotations(rsmProto, c.Cluster)
+ c.workloadVertex.Obj = rsmProto
+ c.workloadVertex.Action = ictrltypes.ActionCreatePtr()
+}
+
+func (c *rsmComponent) updateWorkload(rsmObj *workloads.ReplicatedStateMachine) bool {
+ rsmObjCopy := rsmObj.DeepCopy()
+ rsmProto := c.workloadVertex.Obj.(*workloads.ReplicatedStateMachine)
+
+ // remove original monitor annotations
+ if len(rsmObjCopy.Annotations) > 0 {
+ maps.DeleteFunc(rsmObjCopy.Annotations, func(k, v string) bool {
+ return strings.HasPrefix(k, "monitor.kubeblocks.io")
+ })
+ }
+ mergeAnnotations(rsmObjCopy.Annotations, &rsmProto.Annotations)
+ rsmObjCopy.Annotations = rsmProto.Annotations
+ buildWorkLoadAnnotations(rsmObjCopy, c.Cluster)
+
+ // keep the original template annotations.
+ // if annotations exist and are replaced, the rsm will be updated.
+ mergeAnnotations(rsmObjCopy.Spec.Template.Annotations, &rsmProto.Spec.Template.Annotations)
+ rsmObjCopy.Spec.Template = rsmProto.Spec.Template
+ rsmObjCopy.Spec.Replicas = rsmProto.Spec.Replicas
+ c.updateUpdateStrategy(rsmObjCopy, rsmProto)
+ rsmObjCopy.Spec.Service = rsmProto.Spec.Service
+ rsmObjCopy.Spec.AlternativeServices = rsmProto.Spec.AlternativeServices
+ rsmObjCopy.Spec.Roles = rsmProto.Spec.Roles
+ rsmObjCopy.Spec.RoleProbe = rsmProto.Spec.RoleProbe
+ rsmObjCopy.Spec.MembershipReconfiguration = rsmProto.Spec.MembershipReconfiguration
+ rsmObjCopy.Spec.MemberUpdateStrategy = rsmProto.Spec.MemberUpdateStrategy
+ rsmObjCopy.Spec.Credential = rsmProto.Spec.Credential
+
+ resolvePodSpecDefaultFields(rsmObj.Spec.Template.Spec, &rsmObjCopy.Spec.Template.Spec)
+
+ delayUpdatePodSpecSystemFields(rsmObj.Spec.Template.Spec, &rsmObjCopy.Spec.Template.Spec)
+ isTemplateUpdated := !reflect.DeepEqual(&rsmObj.Spec, &rsmObjCopy.Spec)
+ if isTemplateUpdated {
+ updatePodSpecSystemFields(&rsmObjCopy.Spec.Template.Spec)
+ }
+ if isTemplateUpdated || !reflect.DeepEqual(rsmObj.Annotations, rsmObjCopy.Annotations) {
+ c.workloadVertex.Obj = rsmObjCopy
+ c.workloadVertex.Action = ictrltypes.ActionPtr(ictrltypes.UPDATE)
+ return true
+ }
+ return false
+}
+
+func (c *rsmComponent) updatePDB(reqCtx intctrlutil.RequestCtx, cli client.Client) error {
+ pdbObjList, err := listObjWithLabelsInNamespace(reqCtx.Ctx, cli, generics.PodDisruptionBudgetSignature, c.GetNamespace(), c.getMatchingLabels())
+ if err != nil && !apierrors.IsNotFound(err) {
+ return err
+ }
+ for _, v := range ictrltypes.FindAll[*policyv1.PodDisruptionBudget](c.dag) {
+ node := v.(*ictrltypes.LifecycleVertex)
+ pdbProto := node.Obj.(*policyv1.PodDisruptionBudget)
+
+ if pos := slices.IndexFunc(pdbObjList, func(pdbObj *policyv1.PodDisruptionBudget) bool {
+ return pdbObj.GetName() == pdbProto.GetName()
+ }); pos < 0 {
+ node.Action = ictrltypes.ActionCreatePtr() // TODO: Create or Noop?
+ } else {
+ pdbObj := pdbObjList[pos]
+ if !reflect.DeepEqual(pdbObj.Spec, pdbProto.Spec) {
+ pdbObj.Spec = pdbProto.Spec
+ node.Obj = pdbObj
+ node.Action = ictrltypes.ActionUpdatePtr()
+ }
+ }
+ }
+ return nil
+}
+
+func (c *rsmComponent) updateUpdateStrategy(rsmObj, rsmProto *workloads.ReplicatedStateMachine) {
+ var objMaxUnavailable *intstr.IntOrString
+ if rsmObj.Spec.UpdateStrategy.RollingUpdate != nil {
+ objMaxUnavailable = rsmObj.Spec.UpdateStrategy.RollingUpdate.MaxUnavailable
+ }
+ rsmObj.Spec.UpdateStrategy = rsmProto.Spec.UpdateStrategy
+ if objMaxUnavailable == nil && rsmObj.Spec.UpdateStrategy.RollingUpdate != nil {
+ // HACK: This field is alpha-level (since v1.24) and is only honored by servers that enable the
+ // MaxUnavailableStatefulSet feature.
+ // When we get a nil MaxUnavailable from k8s, we consider that the field is not supported by the server,
+ // and set the MaxUnavailable as nil explicitly to avoid the workload been updated unexpectedly.
+ // Ref: https://kubernetes.io/docs/concepts/workloads/controllers/statefulset/#maximum-unavailable-pods
+ rsmObj.Spec.UpdateStrategy.RollingUpdate.MaxUnavailable = nil
+ }
+}
+
+func (c *rsmComponent) updateVolumes(reqCtx intctrlutil.RequestCtx, cli client.Client, rsmObj *workloads.ReplicatedStateMachine) error {
+ // PVCs which have been added to the dag because of volume expansion.
+ pvcNameSet := sets.New[string]()
+ for _, v := range ictrltypes.FindAll[*corev1.PersistentVolumeClaim](c.dag) {
+ pvcNameSet.Insert(v.(*ictrltypes.LifecycleVertex).Obj.GetName())
}
- synthesizedComp, err := component.BuildComponent(reqCtx, clsMgr, cluster, definition, compDef, compSpec, serviceReferences, compVer)
+ for _, vct := range c.component.VolumeClaimTemplates {
+ pvcs, err := c.getRunningVolumes(reqCtx, cli, vct.Name, rsmObj)
+ if err != nil {
+ return err
+ }
+ for _, pvc := range pvcs {
+ if pvcNameSet.Has(pvc.Name) {
+ continue
+ }
+ c.noopResource(pvc, c.workloadVertex)
+ }
+ }
+ return nil
+}
+
+func (c *rsmComponent) getRunningVolumes(reqCtx intctrlutil.RequestCtx, cli client.Client, vctName string,
+ rsmObj *workloads.ReplicatedStateMachine) ([]*corev1.PersistentVolumeClaim, error) {
+ pvcs, err := listObjWithLabelsInNamespace(reqCtx.Ctx, cli, generics.PersistentVolumeClaimSignature, c.GetNamespace(), c.getMatchingLabels())
if err != nil {
+ if apierrors.IsNotFound(err) {
+ return nil, nil
+ }
return nil, err
}
- if synthesizedComp == nil {
- return nil, nil
+ matchedPVCs := make([]*corev1.PersistentVolumeClaim, 0)
+ prefix := fmt.Sprintf("%s-%s", vctName, rsmObj.Name)
+ for _, pvc := range pvcs {
+ if strings.HasPrefix(pvc.Name, prefix) {
+ matchedPVCs = append(matchedPVCs, pvc)
+ }
}
+ return matchedPVCs, nil
+}
- if intctrlutil.IsRSMEnabled() {
- return newRSMComponent(cli, reqCtx.Recorder, cluster, version, synthesizedComp, dag), nil
+// hasFailedAndTimedOutPod returns whether the pods of components are still failed after a PodFailedTimeout period.
+func hasFailedAndTimedOutPod(pods []*corev1.Pod) (bool, appsv1alpha1.ComponentMessageMap, time.Duration) {
+ var (
+ hasTimedOutPod bool
+ messages = appsv1alpha1.ComponentMessageMap{}
+ hasFailedPod bool
+ requeueAfter time.Duration
+ )
+ for _, pod := range pods {
+ isFailed, isTimedOut, messageStr := isPodFailedAndTimedOut(pod)
+ if !isFailed {
+ continue
+ }
+ if isTimedOut {
+ hasTimedOutPod = true
+ messages.SetObjectMessage(pod.Kind, pod.Name, messageStr)
+ } else {
+ hasFailedPod = true
+ }
+ }
+ if hasFailedPod && !hasTimedOutPod {
+ requeueAfter = podContainerFailedTimeout
}
+ return hasTimedOutPod, messages, requeueAfter
+}
- switch compDef.WorkloadType {
- case appsv1alpha1.Replication:
- return newReplicationComponent(cli, reqCtx.Recorder, cluster, version, synthesizedComp, dag), nil
- case appsv1alpha1.Consensus:
- return newConsensusComponent(cli, reqCtx.Recorder, cluster, version, synthesizedComp, dag), nil
- case appsv1alpha1.Stateful:
- return newStatefulComponent(cli, reqCtx.Recorder, cluster, version, synthesizedComp, dag), nil
- case appsv1alpha1.Stateless:
- return newStatelessComponent(cli, reqCtx.Recorder, cluster, version, synthesizedComp, dag), nil
+// isPodScheduledFailedAndTimedOut checks whether the unscheduled pod has timed out.
+func isPodScheduledFailedAndTimedOut(pod *corev1.Pod) (bool, bool, string) {
+ for _, cond := range pod.Status.Conditions {
+ if cond.Type != corev1.PodScheduled {
+ continue
+ }
+ if cond.Status == corev1.ConditionTrue {
+ return false, false, ""
+ }
+ return true, time.Now().After(cond.LastTransitionTime.Add(podScheduledFailedTimeout)), cond.Message
+ }
+ return false, false, ""
+}
+
+// isPodFailedAndTimedOut checks if the pod is failed and timed out.
+func isPodFailedAndTimedOut(pod *corev1.Pod) (bool, bool, string) {
+ if isFailed, isTimedOut, message := isPodScheduledFailedAndTimedOut(pod); isFailed {
+ return isFailed, isTimedOut, message
+ }
+ initContainerFailed, message := isAnyContainerFailed(pod.Status.InitContainerStatuses)
+ if initContainerFailed {
+ return initContainerFailed, isContainerFailedAndTimedOut(pod, corev1.PodInitialized), message
+ }
+ containerFailed, message := isAnyContainerFailed(pod.Status.ContainerStatuses)
+ if containerFailed {
+ return containerFailed, isContainerFailedAndTimedOut(pod, corev1.ContainersReady), message
+ }
+ return false, false, ""
+}
+
+// isAnyContainerFailed checks whether any container in the list is failed.
+func isAnyContainerFailed(containersStatus []corev1.ContainerStatus) (bool, string) {
+ for _, v := range containersStatus {
+ waitingState := v.State.Waiting
+ if waitingState != nil && waitingState.Message != "" {
+ return true, waitingState.Message
+ }
+ terminatedState := v.State.Terminated
+ if terminatedState != nil && terminatedState.Message != "" {
+ return true, terminatedState.Message
+ }
}
- panic(fmt.Sprintf("unknown workload type: %s, cluster: %s, component: %s, component definition ref: %s",
- compDef.WorkloadType, cluster.Name, compSpec.Name, compSpec.ComponentDefRef))
+ return false, ""
}
-func getClassManager(ctx context.Context, cli types2.ReadonlyClient, cluster *appsv1alpha1.Cluster) (*class.Manager, error) {
- var classDefinitionList appsv1alpha1.ComponentClassDefinitionList
- ml := []client.ListOption{
- client.MatchingLabels{constant.ClusterDefLabelKey: cluster.Spec.ClusterDefRef},
+// isContainerFailedAndTimedOut checks whether the failed container has timed out.
+func isContainerFailedAndTimedOut(pod *corev1.Pod, podConditionType corev1.PodConditionType) bool {
+ containerReadyCondition := intctrlutil.GetPodCondition(&pod.Status, podConditionType)
+ if containerReadyCondition == nil || containerReadyCondition.LastTransitionTime.IsZero() {
+ return false
}
- if err := cli.List(ctx, &classDefinitionList, ml...); err != nil {
+ return time.Now().After(containerReadyCondition.LastTransitionTime.Add(podContainerFailedTimeout))
+}
+
+type gvkName struct {
+ gvk schema.GroupVersionKind
+ ns, name string
+}
+
+type clusterSnapshot map[gvkName]client.Object
+
+func getGVKName(object client.Object, scheme *runtime.Scheme) (*gvkName, error) {
+ gvk, err := apiutil.GVKForObject(object, scheme)
+ if err != nil {
return nil, err
}
+ return &gvkName{
+ gvk: gvk,
+ ns: object.GetNamespace(),
+ name: object.GetName(),
+ }, nil
+}
- var constraintList appsv1alpha1.ComponentResourceConstraintList
- if err := cli.List(ctx, &constraintList); err != nil {
+func isOwnerOf(owner, obj client.Object, scheme *runtime.Scheme) bool {
+ ro, ok := owner.(runtime.Object)
+ if !ok {
+ return false
+ }
+ gvk, err := apiutil.GVKForObject(ro, scheme)
+ if err != nil {
+ return false
+ }
+ ref := metav1.OwnerReference{
+ APIVersion: gvk.GroupVersion().String(),
+ Kind: gvk.Kind,
+ UID: owner.GetUID(),
+ Name: owner.GetName(),
+ }
+ owners := obj.GetOwnerReferences()
+ referSameObject := func(a, b metav1.OwnerReference) bool {
+ aGV, err := schema.ParseGroupVersion(a.APIVersion)
+ if err != nil {
+ return false
+ }
+
+ bGV, err := schema.ParseGroupVersion(b.APIVersion)
+ if err != nil {
+ return false
+ }
+
+ return aGV.Group == bGV.Group && a.Kind == b.Kind && a.Name == b.Name
+ }
+ for _, ownerRef := range owners {
+ if referSameObject(ownerRef, ref) {
+ return true
+ }
+ }
+ return false
+}
+
+func ownedKinds() []client.ObjectList {
+ return []client.ObjectList{
+ &appsv1.StatefulSetList{},
+ &appsv1.DeploymentList{},
+ &corev1.ServiceList{},
+ &corev1.SecretList{},
+ &corev1.ConfigMapList{},
+ &corev1.PersistentVolumeClaimList{}, // TODO(merge): remove it?
+ &policyv1.PodDisruptionBudgetList{},
+ &dataprotectionv1alpha1.BackupPolicyList{},
+ }
+}
+
+// read all objects owned by component
+func readCacheSnapshot(reqCtx intctrlutil.RequestCtx, cli client.Client, cluster *appsv1alpha1.Cluster) (clusterSnapshot, error) {
+ // list what kinds of object cluster owns
+ kinds := ownedKinds()
+ snapshot := make(clusterSnapshot)
+ ml := client.MatchingLabels{constant.AppInstanceLabelKey: cluster.GetName()}
+ inNS := client.InNamespace(cluster.Namespace)
+ for _, list := range kinds {
+ if err := cli.List(reqCtx.Ctx, list, inNS, ml); err != nil {
+ return nil, err
+ }
+ // reflect get list.Items
+ items := reflect.ValueOf(list).Elem().FieldByName("Items")
+ l := items.Len()
+ for i := 0; i < l; i++ {
+ // get the underlying object
+ object := items.Index(i).Addr().Interface().(client.Object)
+ // put to snapshot if owned by our cluster
+ if isOwnerOf(cluster, object, cli.Scheme()) {
+ name, err := getGVKName(object, cli.Scheme())
+ if err != nil {
+ return nil, err
+ }
+ snapshot[*name] = object
+ }
+ }
+ }
+ return snapshot, nil
+}
+
+func resolveObjectAction(snapshot clusterSnapshot, vertex *ictrltypes.LifecycleVertex, scheme *runtime.Scheme) (*ictrltypes.LifecycleAction, error) {
+ gvk, err := getGVKName(vertex.Obj, scheme)
+ if err != nil {
return nil, err
}
- return class.NewManager(classDefinitionList, constraintList)
+ if _, ok := snapshot[*gvk]; ok {
+ return ictrltypes.ActionNoopPtr(), nil
+ } else {
+ return ictrltypes.ActionCreatePtr(), nil
+ }
}
diff --git a/controllers/apps/components/component_set.go b/controllers/apps/components/component_set.go
deleted file mode 100644
index bc04c07b40b..00000000000
--- a/controllers/apps/components/component_set.go
+++ /dev/null
@@ -1,71 +0,0 @@
-/*
-Copyright (C) 2022-2023 ApeCloud Co., Ltd
-
-This file is part of KubeBlocks project
-
-This program is free software: you can redistribute it and/or modify
-it under the terms of the GNU Affero General Public License as published by
-the Free Software Foundation, either version 3 of the License, or
-(at your option) any later version.
-
-This program is distributed in the hope that it will be useful
-but WITHOUT ANY WARRANTY; without even the implied warranty of
-MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
-GNU Affero General Public License for more details.
-
-You should have received a copy of the GNU Affero General Public License
-along with this program. If not, see .
-*/
-
-package components
-
-import (
- "context"
-
- corev1 "k8s.io/api/core/v1"
- "sigs.k8s.io/controller-runtime/pkg/client"
-
- "github.com/apecloud/kubeblocks/apis/apps/v1alpha1"
- "github.com/apecloud/kubeblocks/internal/controller/component"
- "github.com/apecloud/kubeblocks/internal/controller/graph"
-)
-
-// TODO(impl): replace it with ComponentWorkload and <*>Set implementation.
-
-type componentSet interface {
- // IsRunning when relevant k8s workloads changes, it checks whether the component is running.
- // you can also reconcile the pods of component till the component is Running here.
- IsRunning(ctx context.Context, obj client.Object) (bool, error)
-
- // PodsReady checks whether all pods of the component are ready.
- // it means the pods are available in StatefulSet or Deployment.
- PodsReady(ctx context.Context, obj client.Object) (bool, error)
-
- // PodIsAvailable checks whether a pod of the component is available.
- // if the component is Stateless/StatefulSet, the available conditions follows as:
- // 1. the pod is ready.
- // 2. readyTime reached minReadySeconds.
- // if the component is consensusSet,it will be available when the pod is ready and labeled with its role.
- PodIsAvailable(pod *corev1.Pod, minReadySeconds int32) bool
-
- // GetPhaseWhenPodsReadyAndProbeTimeout when the pods of component are ready but the probe timed-out,
- // calculate the component phase is Failed or Abnormal.
- GetPhaseWhenPodsReadyAndProbeTimeout(pods []*corev1.Pod) (v1alpha1.ClusterComponentPhase, v1alpha1.ComponentMessageMap)
-
- // GetPhaseWhenPodsNotReady when the pods of component are not ready, calculate the component phase is Failed or Abnormal.
- // if return an empty phase, means the pods of component are ready and skips it.
- GetPhaseWhenPodsNotReady(ctx context.Context, componentName string, originPhaseIsUpRunning bool) (v1alpha1.ClusterComponentPhase, v1alpha1.ComponentMessageMap, error)
-
- HandleRestart(ctx context.Context, obj client.Object) ([]graph.Vertex, error)
-
- HandleRoleChange(ctx context.Context, obj client.Object) ([]graph.Vertex, error)
-}
-
-// componentSetBase is a common component set base struct.
-type componentSetBase struct {
- Cli client.Client
- Cluster *v1alpha1.Cluster
- SynthesizedComponent *component.SynthesizedComponent
- ComponentSpec *v1alpha1.ClusterComponentSpec // for test cases used only
- ComponentDef *v1alpha1.ClusterComponentDefinition // for test cases used only
-}
diff --git a/controllers/apps/components/workload_builder.go b/controllers/apps/components/component_workload_builder.go
similarity index 63%
rename from controllers/apps/components/workload_builder.go
rename to controllers/apps/components/component_workload_builder.go
index 2f319e80a16..6fe93205095 100644
--- a/controllers/apps/components/workload_builder.go
+++ b/controllers/apps/components/component_workload_builder.go
@@ -35,8 +35,6 @@ import (
intctrlutil "github.com/apecloud/kubeblocks/internal/controllerutil"
)
-// TODO(impl): define a custom workload to encapsulate all the resources.
-
type componentWorkloadBuilder interface {
// runtime, config, script, env, volume, service, monitor, probe
BuildEnv() componentWorkloadBuilder
@@ -44,90 +42,83 @@ type componentWorkloadBuilder interface {
BuildWorkload() componentWorkloadBuilder
BuildPDB() componentWorkloadBuilder
BuildVolumeMount() componentWorkloadBuilder
- BuildService() componentWorkloadBuilder
- BuildHeadlessService() componentWorkloadBuilder
BuildTLSCert() componentWorkloadBuilder
BuildTLSVolume() componentWorkloadBuilder
Complete() error
}
-type componentWorkloadBuilderBase struct {
- ReqCtx intctrlutil.RequestCtx
- Client client.Client
- Comp Component
- DefaultAction *ictrltypes.LifecycleAction
- ConcreteBuilder componentWorkloadBuilder
- Error error
- EnvConfig *corev1.ConfigMap
- Workload client.Object
- LocalObjs []client.Object // cache the objects needed for configuration, should remove this after refactoring the configuration
+type rsmComponentWorkloadBuilder struct {
+ reqCtx intctrlutil.RequestCtx
+ client client.Client
+ comp *rsmComponent
+ defaultAction *ictrltypes.LifecycleAction
+ error error
+ envConfig *corev1.ConfigMap
+ workload client.Object
+ localObjs []client.Object // cache the objects needed for configuration, should remove this after refactoring the configuration
}
-func (b *componentWorkloadBuilderBase) BuildEnv() componentWorkloadBuilder {
+var _ componentWorkloadBuilder = &rsmComponentWorkloadBuilder{}
+
+func (b *rsmComponentWorkloadBuilder) BuildEnv() componentWorkloadBuilder {
buildfn := func() ([]client.Object, error) {
- envCfg, err := factory.BuildEnvConfig(b.Comp.GetCluster(), b.Comp.GetSynthesizedComponent())
- b.EnvConfig = envCfg
- b.LocalObjs = append(b.LocalObjs, envCfg)
- return []client.Object{envCfg}, err
+ envCfg := factory.BuildEnvConfig(b.comp.GetCluster(), b.comp.GetSynthesizedComponent())
+ b.envConfig = envCfg
+ b.localObjs = append(b.localObjs, envCfg)
+ return []client.Object{envCfg}, nil
}
return b.BuildWrapper(buildfn)
}
-func (b *componentWorkloadBuilderBase) BuildConfig() componentWorkloadBuilder {
+func (b *rsmComponentWorkloadBuilder) BuildConfig() componentWorkloadBuilder {
buildfn := func() ([]client.Object, error) {
- if b.Workload == nil {
+ if b.workload == nil {
return nil, fmt.Errorf("build config but workload is nil, cluster: %s, component: %s",
- b.Comp.GetClusterName(), b.Comp.GetName())
+ b.comp.GetClusterName(), b.comp.GetName())
}
err := plan.RenderConfigNScriptFiles(
&intctrlutil.ResourceCtx{
- Context: b.ReqCtx.Ctx,
- Client: b.Client,
- Namespace: b.Comp.GetNamespace(),
- ClusterName: b.Comp.GetClusterName(),
- ComponentName: b.Comp.GetName(),
+ Context: b.reqCtx.Ctx,
+ Client: b.client,
+ Namespace: b.comp.GetNamespace(),
+ ClusterName: b.comp.GetClusterName(),
+ ComponentName: b.comp.GetName(),
},
- b.Comp.GetClusterVersion(),
- b.Comp.GetCluster(),
- b.Comp.GetSynthesizedComponent(),
- b.Workload,
+ b.comp.GetClusterVersion(),
+ b.comp.GetCluster(),
+ b.comp.GetSynthesizedComponent(),
+ b.workload,
b.getRuntime(),
- b.LocalObjs)
+ b.localObjs)
return nil, err
}
return b.BuildWrapper(buildfn)
}
-func (b *componentWorkloadBuilderBase) BuildWorkload4StatefulSet(workloadType string) componentWorkloadBuilder {
+func (b *rsmComponentWorkloadBuilder) BuildWorkload() componentWorkloadBuilder {
buildfn := func() ([]client.Object, error) {
- if b.EnvConfig == nil {
- return nil, fmt.Errorf("build %s workload but env config is nil, cluster: %s, component: %s",
- workloadType, b.Comp.GetClusterName(), b.Comp.GetName())
- }
-
- sts, err := factory.BuildSts(b.ReqCtx, b.Comp.GetCluster(), b.Comp.GetSynthesizedComponent(), b.EnvConfig.Name)
+ component := b.comp.GetSynthesizedComponent()
+ obj, err := factory.BuildRSM(b.reqCtx, b.comp.GetCluster(), component, b.envConfig.Name)
if err != nil {
return nil, err
}
- b.Workload = sts
+
+ b.workload = obj
return nil, nil // don't return sts here
}
return b.BuildWrapper(buildfn)
}
-func (b *componentWorkloadBuilderBase) BuildPDB() componentWorkloadBuilder {
+func (b *rsmComponentWorkloadBuilder) BuildPDB() componentWorkloadBuilder {
buildfn := func() ([]client.Object, error) {
// if without this handler, the cluster controller will occur error during reconciling.
// conditionally build PodDisruptionBudget
- synthesizedComponent := b.Comp.GetSynthesizedComponent()
+ synthesizedComponent := b.comp.GetSynthesizedComponent()
if synthesizedComponent.MinAvailable != nil {
- pdb, err := factory.BuildPDB(b.Comp.GetCluster(), synthesizedComponent)
- if err != nil {
- return nil, err
- }
+ pdb := factory.BuildPDB(b.comp.GetCluster(), synthesizedComponent)
return []client.Object{pdb}, nil
} else {
panic("this shouldn't happen")
@@ -136,11 +127,11 @@ func (b *componentWorkloadBuilderBase) BuildPDB() componentWorkloadBuilder {
return b.BuildWrapper(buildfn)
}
-func (b *componentWorkloadBuilderBase) BuildVolumeMount() componentWorkloadBuilder {
+func (b *rsmComponentWorkloadBuilder) BuildVolumeMount() componentWorkloadBuilder {
buildfn := func() ([]client.Object, error) {
- if b.Workload == nil {
+ if b.workload == nil {
return nil, fmt.Errorf("build volume mount but workload is nil, cluster: %s, component: %s",
- b.Comp.GetClusterName(), b.Comp.GetName())
+ b.comp.GetClusterName(), b.comp.GetName())
}
podSpec := b.getRuntime()
@@ -167,33 +158,10 @@ func (b *componentWorkloadBuilderBase) BuildVolumeMount() componentWorkloadBuild
return b.BuildWrapper(buildfn)
}
-func (b *componentWorkloadBuilderBase) BuildService() componentWorkloadBuilder {
- buildfn := func() ([]client.Object, error) {
- svcList, err := factory.BuildSvcList(b.Comp.GetCluster(), b.Comp.GetSynthesizedComponent())
- if err != nil {
- return nil, err
- }
- objs := make([]client.Object, 0)
- for _, svc := range svcList {
- objs = append(objs, svc)
- }
- return objs, err
- }
- return b.BuildWrapper(buildfn)
-}
-
-func (b *componentWorkloadBuilderBase) BuildHeadlessService() componentWorkloadBuilder {
- buildfn := func() ([]client.Object, error) {
- svc, err := factory.BuildHeadlessSvc(b.Comp.GetCluster(), b.Comp.GetSynthesizedComponent())
- return []client.Object{svc}, err
- }
- return b.BuildWrapper(buildfn)
-}
-
-func (b *componentWorkloadBuilderBase) BuildTLSCert() componentWorkloadBuilder {
+func (b *rsmComponentWorkloadBuilder) BuildTLSCert() componentWorkloadBuilder {
buildfn := func() ([]client.Object, error) {
- cluster := b.Comp.GetCluster()
- component := b.Comp.GetSynthesizedComponent()
+ cluster := b.comp.GetCluster()
+ component := b.comp.GetSynthesizedComponent()
if !component.TLS {
return nil, nil
}
@@ -204,7 +172,7 @@ func (b *componentWorkloadBuilderBase) BuildTLSCert() componentWorkloadBuilder {
objs := make([]client.Object, 0)
switch component.Issuer.Name {
case appsv1alpha1.IssuerUserProvided:
- if err := plan.CheckTLSSecretRef(b.ReqCtx.Ctx, b.Client, cluster.Namespace, component.Issuer.SecretRef); err != nil {
+ if err := plan.CheckTLSSecretRef(b.reqCtx.Ctx, b.client, cluster.Namespace, component.Issuer.SecretRef); err != nil {
return nil, err
}
case appsv1alpha1.IssuerKubeBlocks:
@@ -213,59 +181,59 @@ func (b *componentWorkloadBuilderBase) BuildTLSCert() componentWorkloadBuilder {
return nil, err
}
objs = append(objs, secret)
- b.LocalObjs = append(b.LocalObjs, secret)
+ b.localObjs = append(b.localObjs, secret)
}
return objs, nil
}
return b.BuildWrapper(buildfn)
}
-func (b *componentWorkloadBuilderBase) BuildTLSVolume() componentWorkloadBuilder {
+func (b *rsmComponentWorkloadBuilder) BuildTLSVolume() componentWorkloadBuilder {
buildfn := func() ([]client.Object, error) {
- if b.Workload == nil {
+ if b.workload == nil {
return nil, fmt.Errorf("build TLS volumes but workload is nil, cluster: %s, component: %s",
- b.Comp.GetClusterName(), b.Comp.GetName())
+ b.comp.GetClusterName(), b.comp.GetName())
}
// build secret volume and volume mount
- return nil, updateTLSVolumeAndVolumeMount(b.getRuntime(), b.Comp.GetClusterName(), *b.Comp.GetSynthesizedComponent())
+ return nil, updateTLSVolumeAndVolumeMount(b.getRuntime(), b.comp.GetClusterName(), *b.comp.GetSynthesizedComponent())
}
return b.BuildWrapper(buildfn)
}
-func (b *componentWorkloadBuilderBase) Complete() error {
- if b.Error != nil {
- return b.Error
+func (b *rsmComponentWorkloadBuilder) Complete() error {
+ if b.error != nil {
+ return b.error
}
- if b.Workload == nil {
+ if b.workload == nil {
return fmt.Errorf("fail to create component workloads, cluster: %s, component: %s",
- b.Comp.GetClusterName(), b.Comp.GetName())
+ b.comp.GetClusterName(), b.comp.GetName())
}
- b.Comp.SetWorkload(b.Workload, b.DefaultAction, nil)
+ b.comp.setWorkload(b.workload, b.defaultAction, nil)
return nil
}
-func (b *componentWorkloadBuilderBase) BuildWrapper(buildfn func() ([]client.Object, error)) componentWorkloadBuilder {
- if b.Error != nil || buildfn == nil {
- return b.ConcreteBuilder
+func (b *rsmComponentWorkloadBuilder) BuildWrapper(buildfn func() ([]client.Object, error)) componentWorkloadBuilder {
+ if b.error != nil || buildfn == nil {
+ return b
}
objs, err := buildfn()
if err != nil {
- b.Error = err
+ b.error = err
} else {
- cluster := b.Comp.GetCluster()
- component := b.Comp.GetSynthesizedComponent()
+ cluster := b.comp.GetCluster()
+ component := b.comp.GetSynthesizedComponent()
if err = updateCustomLabelToObjs(cluster.Name, string(cluster.UID), component.Name, component.CustomLabelSpecs, objs); err != nil {
- b.Error = err
+ b.error = err
}
for _, obj := range objs {
- b.Comp.AddResource(obj, b.DefaultAction, nil)
+ b.comp.addResource(obj, b.defaultAction, nil)
}
}
- return b.ConcreteBuilder
+ return b
}
-func (b *componentWorkloadBuilderBase) getRuntime() *corev1.PodSpec {
- switch w := b.Workload.(type) {
+func (b *rsmComponentWorkloadBuilder) getRuntime() *corev1.PodSpec {
+ switch w := b.workload.(type) {
case *appsv1.StatefulSet:
return &w.Spec.Template.Spec
case *appsv1.Deployment:
diff --git a/controllers/apps/components/consensus.go b/controllers/apps/components/consensus.go
deleted file mode 100644
index 7744d821f17..00000000000
--- a/controllers/apps/components/consensus.go
+++ /dev/null
@@ -1,107 +0,0 @@
-/*
-Copyright (C) 2022-2023 ApeCloud Co., Ltd
-
-This file is part of KubeBlocks project
-
-This program is free software: you can redistribute it and/or modify
-it under the terms of the GNU Affero General Public License as published by
-the Free Software Foundation, either version 3 of the License, or
-(at your option) any later version.
-
-This program is distributed in the hope that it will be useful
-but WITHOUT ANY WARRANTY; without even the implied warranty of
-MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
-GNU Affero General Public License for more details.
-
-You should have received a copy of the GNU Affero General Public License
-along with this program. If not, see .
-*/
-
-package components
-
-import (
- "k8s.io/client-go/tools/record"
- "sigs.k8s.io/controller-runtime/pkg/client"
-
- appsv1alpha1 "github.com/apecloud/kubeblocks/apis/apps/v1alpha1"
- "github.com/apecloud/kubeblocks/internal/controller/component"
- "github.com/apecloud/kubeblocks/internal/controller/graph"
- ictrltypes "github.com/apecloud/kubeblocks/internal/controller/types"
- intctrlutil "github.com/apecloud/kubeblocks/internal/controllerutil"
-)
-
-func newConsensusComponent(cli client.Client,
- recorder record.EventRecorder,
- cluster *appsv1alpha1.Cluster,
- clusterVersion *appsv1alpha1.ClusterVersion,
- synthesizedComponent *component.SynthesizedComponent,
- dag *graph.DAG) *consensusComponent {
- comp := &consensusComponent{
- statefulComponentBase: statefulComponentBase{
- componentBase: componentBase{
- Client: cli,
- Recorder: recorder,
- Cluster: cluster,
- ClusterVersion: clusterVersion,
- Component: synthesizedComponent,
- ComponentSet: &consensusSet{
- stateful: stateful{
- componentSetBase: componentSetBase{
- Cli: cli,
- Cluster: cluster,
- SynthesizedComponent: synthesizedComponent,
- ComponentSpec: nil,
- ComponentDef: nil,
- },
- },
- },
- Dag: dag,
- WorkloadVertex: nil,
- },
- },
- }
- return comp
-}
-
-type consensusComponent struct {
- statefulComponentBase
-}
-
-var _ Component = &consensusComponent{}
-
-func (c *consensusComponent) newBuilder(reqCtx intctrlutil.RequestCtx, cli client.Client,
- action *ictrltypes.LifecycleAction) componentWorkloadBuilder {
- builder := &consensusComponentWorkloadBuilder{
- componentWorkloadBuilderBase: componentWorkloadBuilderBase{
- ReqCtx: reqCtx,
- Client: cli,
- Comp: c,
- DefaultAction: action,
- Error: nil,
- EnvConfig: nil,
- Workload: nil,
- },
- }
- builder.ConcreteBuilder = builder
- return builder
-}
-
-func (c *consensusComponent) GetWorkloadType() appsv1alpha1.WorkloadType {
- return appsv1alpha1.Consensus
-}
-
-func (c *consensusComponent) GetBuiltObjects(reqCtx intctrlutil.RequestCtx, cli client.Client) ([]client.Object, error) {
- return c.statefulComponentBase.GetBuiltObjects(c.newBuilder(reqCtx, cli, ictrltypes.ActionCreatePtr()))
-}
-
-func (c *consensusComponent) Create(reqCtx intctrlutil.RequestCtx, cli client.Client) error {
- return c.statefulComponentBase.Create(reqCtx, cli, c.newBuilder(reqCtx, cli, ictrltypes.ActionCreatePtr()))
-}
-
-func (c *consensusComponent) Update(reqCtx intctrlutil.RequestCtx, cli client.Client) error {
- return c.statefulComponentBase.Update(reqCtx, cli, c.newBuilder(reqCtx, cli, nil))
-}
-
-func (c *consensusComponent) Status(reqCtx intctrlutil.RequestCtx, cli client.Client) error {
- return c.statefulComponentBase.Status(reqCtx, cli, c.newBuilder(reqCtx, cli, ictrltypes.ActionNoopPtr()))
-}
diff --git a/controllers/apps/components/consensus_set.go b/controllers/apps/components/consensus_set.go
deleted file mode 100644
index 4feeae186ce..00000000000
--- a/controllers/apps/components/consensus_set.go
+++ /dev/null
@@ -1,270 +0,0 @@
-/*
-Copyright (C) 2022-2023 ApeCloud Co., Ltd
-
-This file is part of KubeBlocks project
-
-This program is free software: you can redistribute it and/or modify
-it under the terms of the GNU Affero General Public License as published by
-the Free Software Foundation, either version 3 of the License, or
-(at your option) any later version.
-
-This program is distributed in the hope that it will be useful
-but WITHOUT ANY WARRANTY; without even the implied warranty of
-MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
-GNU Affero General Public License for more details.
-
-You should have received a copy of the GNU Affero General Public License
-along with this program. If not, see .
-*/
-
-package components
-
-import (
- "context"
-
- "github.com/google/go-cmp/cmp"
- appsv1 "k8s.io/api/apps/v1"
- corev1 "k8s.io/api/core/v1"
- "sigs.k8s.io/controller-runtime/pkg/client"
-
- appsv1alpha1 "github.com/apecloud/kubeblocks/apis/apps/v1alpha1"
- "github.com/apecloud/kubeblocks/internal/constant"
- "github.com/apecloud/kubeblocks/internal/controller/graph"
- intctrlutil "github.com/apecloud/kubeblocks/internal/controllerutil"
-)
-
-type consensusSet struct {
- stateful
-}
-
-var _ componentSet = &consensusSet{}
-
-func (r *consensusSet) getName() string {
- if r.SynthesizedComponent != nil {
- return r.SynthesizedComponent.Name
- }
- return r.ComponentSpec.Name
-}
-
-func (r *consensusSet) getDefName() string {
- if r.SynthesizedComponent != nil {
- return r.SynthesizedComponent.ComponentDef
- }
- return r.ComponentDef.Name
-}
-
-func (r *consensusSet) getWorkloadType() appsv1alpha1.WorkloadType {
- return appsv1alpha1.Consensus
-}
-
-func (r *consensusSet) getReplicas() int32 {
- if r.SynthesizedComponent != nil {
- return r.SynthesizedComponent.Replicas
- }
- return r.ComponentSpec.Replicas
-}
-
-func (r *consensusSet) getConsensusSpec() *appsv1alpha1.ConsensusSetSpec {
- if r.SynthesizedComponent != nil {
- return r.SynthesizedComponent.ConsensusSpec
- }
- return r.ComponentDef.ConsensusSpec
-}
-
-func (r *consensusSet) getProbes() *appsv1alpha1.ClusterDefinitionProbes {
- if r.SynthesizedComponent != nil {
- return r.SynthesizedComponent.Probes
- }
- return r.ComponentDef.Probes
-}
-
-func (r *consensusSet) IsRunning(ctx context.Context, obj client.Object) (bool, error) {
- if obj == nil {
- return false, nil
- }
- sts := convertToStatefulSet(obj)
- isRevisionConsistent, err := isStsAndPodsRevisionConsistent(ctx, r.Cli, sts)
- if err != nil {
- return false, err
- }
- pods, err := GetPodListByStatefulSet(ctx, r.Cli, sts)
- if err != nil {
- return false, err
- }
- for _, pod := range pods {
- if !intctrlutil.PodIsReadyWithLabel(pod) {
- return false, nil
- }
- }
-
- targetReplicas := r.getReplicas()
- return statefulSetOfComponentIsReady(sts, isRevisionConsistent, &targetReplicas), nil
-}
-
-func (r *consensusSet) PodsReady(ctx context.Context, obj client.Object) (bool, error) {
- return r.stateful.PodsReady(ctx, obj)
-}
-
-func (r *consensusSet) PodIsAvailable(pod *corev1.Pod, minReadySeconds int32) bool {
- if pod == nil {
- return false
- }
- return intctrlutil.PodIsReadyWithLabel(*pod)
-}
-
-func (r *consensusSet) GetPhaseWhenPodsReadyAndProbeTimeout(pods []*corev1.Pod) (appsv1alpha1.ClusterComponentPhase, appsv1alpha1.ComponentMessageMap) {
- var (
- isAbnormal bool
- isFailed = true
- statusMessages appsv1alpha1.ComponentMessageMap
- )
- compStatus, ok := r.Cluster.Status.Components[r.getName()]
- if !ok || compStatus.PodsReadyTime == nil {
- return "", nil
- }
- if !isProbeTimeout(r.getProbes(), compStatus.PodsReadyTime) {
- return "", nil
- }
- for _, pod := range pods {
- role := pod.Labels[constant.RoleLabelKey]
- if role == r.getConsensusSpec().Leader.Name {
- isFailed = false
- }
- if role == "" {
- isAbnormal = true
- statusMessages.SetObjectMessage(pod.Kind, pod.Name, "Role probe timeout, check whether the application is available")
- }
- // TODO clear up the message of ready pod in component.message.
- }
- if isFailed {
- return appsv1alpha1.FailedClusterCompPhase, statusMessages
- }
- if isAbnormal {
- return appsv1alpha1.AbnormalClusterCompPhase, statusMessages
- }
- return "", statusMessages
-}
-
-func (r *consensusSet) GetPhaseWhenPodsNotReady(ctx context.Context,
- componentName string,
- originPhaseIsUpRunning bool) (appsv1alpha1.ClusterComponentPhase, appsv1alpha1.ComponentMessageMap, error) {
- stsList := &appsv1.StatefulSetList{}
- podList, err := getCompRelatedObjectList(ctx, r.Cli, *r.Cluster,
- componentName, stsList)
- if err != nil || len(stsList.Items) == 0 {
- return "", nil, err
- }
- stsObj := stsList.Items[0]
- podCount := len(podList.Items)
- componentReplicas := r.getReplicas()
- if podCount == 0 || stsObj.Status.AvailableReplicas == 0 {
- return getPhaseWithNoAvailableReplicas(componentReplicas), nil, nil
- }
- // get the statefulSet of component
- var (
- existLatestRevisionFailedPod bool
- leaderIsReady bool
- consensusSpec = r.getConsensusSpec()
- statusMessages = appsv1alpha1.ComponentMessageMap{}
- )
- for _, v := range podList.Items {
- // if the pod is terminating, ignore it
- if v.DeletionTimestamp != nil {
- return "", nil, nil
- }
- labelValue := v.Labels[constant.RoleLabelKey]
- if consensusSpec != nil && labelValue == consensusSpec.Leader.Name && intctrlutil.PodIsReady(&v) {
- leaderIsReady = true
- continue
- }
- // if component is up running but pod is not ready, this pod should be failed.
- // for example: full disk cause readiness probe failed and serve is not available.
- // but kubelet only sets the container is not ready and pod is also Running.
- if originPhaseIsUpRunning && !intctrlutil.PodIsReady(&v) && intctrlutil.PodIsControlledByLatestRevision(&v, &stsObj) {
- existLatestRevisionFailedPod = true
- continue
- }
- isFailed, _, message := IsPodFailedAndTimedOut(&v)
- if isFailed && intctrlutil.PodIsControlledByLatestRevision(&v, &stsObj) {
- existLatestRevisionFailedPod = true
- statusMessages.SetObjectMessage(v.Kind, v.Name, message)
- }
- }
- return getCompPhaseByConditions(existLatestRevisionFailedPod, leaderIsReady,
- componentReplicas, int32(podCount), stsObj.Status.AvailableReplicas), statusMessages, nil
-}
-
-func (r *consensusSet) HandleRestart(ctx context.Context, obj client.Object) ([]graph.Vertex, error) {
- if r.getWorkloadType() != appsv1alpha1.Consensus {
- return nil, nil
- }
- priorityMapperFn := func(component *appsv1alpha1.ClusterComponentDefinition) map[string]int {
- return ComposeRolePriorityMap(component.ConsensusSpec)
- }
- return r.HandleUpdateWithStrategy(ctx, obj, nil, priorityMapperFn, generateConsensusSerialPlan, generateConsensusBestEffortParallelPlan, generateConsensusParallelPlan)
-}
-
-// HandleRoleChange is the implementation of the type Component interface method, which is used to handle the role change of the Consensus workload.
-func (r *consensusSet) HandleRoleChange(ctx context.Context, obj client.Object) ([]graph.Vertex, error) {
- if r.getWorkloadType() != appsv1alpha1.Consensus {
- return nil, nil
- }
-
- stsObj := convertToStatefulSet(obj)
- pods, err := GetPodListByStatefulSet(ctx, r.Cli, stsObj)
- if err != nil {
- return nil, err
- }
-
- // update cluster.status.component.consensusSetStatus based on the existences for all pods
- componentName := r.getName()
-
- // first, get the old status
- var oldConsensusSetStatus *appsv1alpha1.ConsensusSetStatus
- if v, ok := r.Cluster.Status.Components[componentName]; ok {
- oldConsensusSetStatus = v.ConsensusSetStatus
- }
- // create the initial status
- newConsensusSetStatus := &appsv1alpha1.ConsensusSetStatus{
- Leader: appsv1alpha1.ConsensusMemberStatus{
- Name: "",
- Pod: constant.ComponentStatusDefaultPodName,
- AccessMode: appsv1alpha1.None,
- },
- }
- // then, set the new status
- setConsensusSetStatusRoles(newConsensusSetStatus, r.getConsensusSpec(), pods)
- // if status changed, do update
- if !cmp.Equal(newConsensusSetStatus, oldConsensusSetStatus) {
- if err = initClusterComponentStatusIfNeed(r.Cluster, componentName, r.getWorkloadType()); err != nil {
- return nil, err
- }
- componentStatus := r.Cluster.Status.Components[componentName]
- componentStatus.ConsensusSetStatus = newConsensusSetStatus
- r.Cluster.Status.SetComponentStatus(componentName, componentStatus)
-
- // TODO: does the update order between cluster and env configmap matter?
-
- // add consensus role info to pod env
- return updateConsensusRoleInfo(ctx, r.Cli, r.Cluster, r.getConsensusSpec(), r.getDefName(), componentName, pods)
- }
- return nil, nil
-}
-
-// newConsensusSet is the constructor of the type consensusSet.
-func newConsensusSet(cli client.Client,
- cluster *appsv1alpha1.Cluster,
- spec *appsv1alpha1.ClusterComponentSpec,
- def appsv1alpha1.ClusterComponentDefinition) *consensusSet {
- return &consensusSet{
- stateful: stateful{
- componentSetBase: componentSetBase{
- Cli: cli,
- Cluster: cluster,
- SynthesizedComponent: nil,
- ComponentSpec: spec,
- ComponentDef: &def,
- },
- },
- }
-}
diff --git a/controllers/apps/components/consensus_set_test.go b/controllers/apps/components/consensus_set_test.go
deleted file mode 100644
index e80c1125a44..00000000000
--- a/controllers/apps/components/consensus_set_test.go
+++ /dev/null
@@ -1,168 +0,0 @@
-/*
-Copyright (C) 2022-2023 ApeCloud Co., Ltd
-
-This file is part of KubeBlocks project
-
-This program is free software: you can redistribute it and/or modify
-it under the terms of the GNU Affero General Public License as published by
-the Free Software Foundation, either version 3 of the License, or
-(at your option) any later version.
-
-This program is distributed in the hope that it will be useful
-but WITHOUT ANY WARRANTY; without even the implied warranty of
-MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
-GNU Affero General Public License for more details.
-
-You should have received a copy of the GNU Affero General Public License
-along with this program. If not, see .
-*/
-
-package components
-
-import (
- "fmt"
- "time"
-
- . "github.com/onsi/ginkgo/v2"
- . "github.com/onsi/gomega"
-
- corev1 "k8s.io/api/core/v1"
- metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
- "sigs.k8s.io/controller-runtime/pkg/client"
-
- appsv1alpha1 "github.com/apecloud/kubeblocks/apis/apps/v1alpha1"
- "github.com/apecloud/kubeblocks/internal/constant"
- intctrlutil "github.com/apecloud/kubeblocks/internal/generics"
- testapps "github.com/apecloud/kubeblocks/internal/testutil/apps"
- testk8s "github.com/apecloud/kubeblocks/internal/testutil/k8s"
-)
-
-var _ = Describe("Consensus Component", func() {
- var (
- randomStr = testCtx.GetRandomStr()
- clusterDefName = "mysql-clusterdef-" + randomStr
- clusterVersionName = "mysql-clusterversion-" + randomStr
- clusterName = "mysql-" + randomStr
- )
-
- const (
- consensusCompName = "consensus"
- defaultMinReadySeconds int32 = 10
- revisionID = "6fdd48d9cd"
- )
-
- cleanAll := func() {
- // must wait until resources deleted and no longer exist before the testcases start,
- // otherwise if later it needs to create some new resource objects with the same name,
- // in race conditions, it will find the existence of old objects, resulting failure to
- // create the new objects.
- By("clean resources")
- // delete cluster(and all dependent sub-resources), clusterversion and clusterdef
- testapps.ClearClusterResources(&testCtx)
-
- // clear rest resources
- inNS := client.InNamespace(testCtx.DefaultNamespace)
- ml := client.HasLabels{testCtx.TestObjLabelKey}
- // namespaced resources
- testapps.ClearResources(&testCtx, intctrlutil.StatefulSetSignature, inNS, ml)
- testapps.ClearResources(&testCtx, intctrlutil.PodSignature, inNS, ml, client.GracePeriodSeconds(0))
- }
-
- BeforeEach(cleanAll)
-
- AfterEach(cleanAll)
-
- mockClusterStatusProbeTimeout := func(cluster *appsv1alpha1.Cluster) {
- // mock pods ready in component status and probe timed out
- Expect(testapps.ChangeObjStatus(&testCtx, cluster, func() {
- podsReady := true
- cluster.Status.Components = map[string]appsv1alpha1.ClusterComponentStatus{
- consensusCompName: {
- PodsReady: &podsReady,
- PodsReadyTime: &metav1.Time{Time: time.Now().Add(-10 * time.Minute)},
- },
- }
- })).ShouldNot(HaveOccurred())
-
- Eventually(testapps.CheckObj(&testCtx, client.ObjectKeyFromObject(cluster), func(g Gomega, tmpCluster *appsv1alpha1.Cluster) {
- g.Expect(tmpCluster.Status.Components).ShouldNot(BeEmpty())
- })).Should(Succeed())
- }
-
- Context("Consensus Component test", func() {
- It("Consensus Component test", func() {
- By(" init cluster, statefulSet, pods")
- clusterDef, _, cluster := testapps.InitConsensusMysql(&testCtx, clusterDefName,
- clusterVersionName, clusterName, "consensus", consensusCompName)
-
- sts := testapps.MockConsensusComponentStatefulSet(&testCtx, clusterName, consensusCompName)
- componentName := consensusCompName
- compDefName := cluster.Spec.GetComponentDefRefName(componentName)
- componentDef := clusterDef.GetComponentDefByName(compDefName)
- component := cluster.Spec.GetComponentByName(componentName)
-
- By("test pods are not ready")
- consensusComponent := newConsensusSet(k8sClient, cluster, component, *componentDef)
- sts.Status.AvailableReplicas = *sts.Spec.Replicas - 1
- podsReady, _ := consensusComponent.PodsReady(ctx, sts)
- Expect(podsReady).Should(BeFalse())
-
- By("test pods are ready")
- // mock sts is ready
- Expect(testapps.ChangeObjStatus(&testCtx, sts, func() {
- controllerRevision := fmt.Sprintf("%s-%s-%s", clusterName, consensusCompName, revisionID)
- sts.Status.CurrentRevision = controllerRevision
- sts.Status.UpdateRevision = controllerRevision
- testk8s.MockStatefulSetReady(sts)
- })).Should(Succeed())
-
- podsReady, _ = consensusComponent.PodsReady(ctx, sts)
- Expect(podsReady).Should(BeTrue())
-
- By("test component is running")
- isRunning, _ := consensusComponent.IsRunning(ctx, sts)
- Expect(isRunning).Should(BeFalse())
-
- podName := sts.Name + "-0"
- podList := testapps.MockConsensusComponentPods(&testCtx, sts, clusterName, consensusCompName)
- By("expect for pod is available")
- Expect(consensusComponent.PodIsAvailable(podList[0], defaultMinReadySeconds)).Should(BeTrue())
-
- By("test handle probe timed out")
- mockClusterStatusProbeTimeout(cluster)
- testk8s.DeletePodLabelKey(ctx, testCtx, podName, constant.RoleLabelKey)
- pod := &corev1.Pod{}
- Expect(testCtx.Cli.Get(ctx, client.ObjectKey{Name: podName, Namespace: testCtx.DefaultNamespace}, pod)).Should(Succeed())
- phase, _ := consensusComponent.GetPhaseWhenPodsReadyAndProbeTimeout([]*corev1.Pod{pod})
- Expect(phase).Should(Equal(appsv1alpha1.FailedClusterCompPhase))
-
- By("test component is running")
- isRunning, _ = consensusComponent.IsRunning(ctx, sts)
- Expect(isRunning).Should(BeFalse())
-
- By("should return empty string if pod of component is only not ready when component is not up running")
- Expect(testapps.ChangeObjStatus(&testCtx, pod, func() {
- pod.Status.Conditions = []corev1.PodCondition{}
- })).Should(Succeed())
- phase, _, _ = consensusComponent.GetPhaseWhenPodsNotReady(ctx, consensusCompName, false)
- Expect(string(phase)).Should(Equal(""))
-
- By("expect component phase is Failed when pod of component is not ready and component is up running")
- phase, _, _ = consensusComponent.GetPhaseWhenPodsNotReady(ctx, consensusCompName, true)
- Expect(phase).Should(Equal(appsv1alpha1.FailedClusterCompPhase))
-
- By("expect component phase is Failed when pod of component is failed")
- testk8s.UpdatePodStatusScheduleFailed(ctx, testCtx, podName, testCtx.DefaultNamespace)
- phase, _, _ = consensusComponent.GetPhaseWhenPodsNotReady(ctx, consensusCompName, false)
- Expect(phase).Should(Equal(appsv1alpha1.FailedClusterCompPhase))
-
- By("unready pod is not controlled by latest revision, should return empty string")
- // mock pod is not controlled by latest revision
- Expect(testapps.ChangeObjStatus(&testCtx, sts, func() {
- sts.Status.UpdateRevision = fmt.Sprintf("%s-%s-%s", clusterName, consensusCompName, "6fdd48d9cd1")
- })).Should(Succeed())
- phase, _, _ = consensusComponent.GetPhaseWhenPodsNotReady(ctx, consensusCompName, false)
- Expect(string(phase)).Should(Equal(""))
- })
- })
-})
diff --git a/controllers/apps/components/consensus_set_utils.go b/controllers/apps/components/consensus_set_utils.go
deleted file mode 100644
index ad15b2e8560..00000000000
--- a/controllers/apps/components/consensus_set_utils.go
+++ /dev/null
@@ -1,425 +0,0 @@
-/*
-Copyright (C) 2022-2023 ApeCloud Co., Ltd
-
-This file is part of KubeBlocks project
-
-This program is free software: you can redistribute it and/or modify
-it under the terms of the GNU Affero General Public License as published by
-the Free Software Foundation, either version 3 of the License, or
-(at your option) any later version.
-
-This program is distributed in the hope that it will be useful
-but WITHOUT ANY WARRANTY; without even the implied warranty of
-MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
-GNU Affero General Public License for more details.
-
-You should have received a copy of the GNU Affero General Public License
-along with this program. If not, see .
-*/
-
-package components
-
-import (
- "context"
- "sort"
- "strings"
- "time"
-
- corev1 "k8s.io/api/core/v1"
- "sigs.k8s.io/controller-runtime/pkg/client"
-
- appsv1alpha1 "github.com/apecloud/kubeblocks/apis/apps/v1alpha1"
- "github.com/apecloud/kubeblocks/internal/constant"
- "github.com/apecloud/kubeblocks/internal/controller/graph"
- ictrltypes "github.com/apecloud/kubeblocks/internal/controller/types"
- intctrlutil "github.com/apecloud/kubeblocks/internal/controllerutil"
-)
-
-// +kubebuilder:rbac:groups=core,resources=pods,verbs=get;list;watch;update;patch;delete
-
-type consensusRole string
-
-type consensusMemberExt struct {
- name string
- consensusRole consensusRole
- accessMode appsv1alpha1.AccessMode
- podName string
-}
-
-const (
- roleLeader consensusRole = "Leader"
- roleFollower consensusRole = "Follower"
- roleLearner consensusRole = "Learner"
-)
-
-const (
- leaderPriority = 1 << 5
- followerReadWritePriority = 1 << 4
- followerReadonlyPriority = 1 << 3
- followerNonePriority = 1 << 2
- learnerPriority = 1 << 1
- emptyConsensusPriority = 1 << 0
- // unknownPriority = 0
-)
-
-// unknown & empty & learner & 1/2 followers -> 1/2 followers -> leader
-func generateConsensusBestEffortParallelPlan(plan *Plan, pods []corev1.Pod, rolePriorityMap map[string]int) {
- start := plan.Start
- // append unknown, empty and learner
- index := 0
- for _, pod := range pods {
- role := pod.Labels[constant.RoleLabelKey]
- if rolePriorityMap[role] <= learnerPriority {
- nextStep := &Step{}
- nextStep.Obj = pod
- start.NextSteps = append(start.NextSteps, nextStep)
- index++
- }
- }
- if len(start.NextSteps) > 0 {
- start = start.NextSteps[0]
- }
- // append 1/2 followers
- podList := pods[index:]
- followerCount := 0
- for _, pod := range podList {
- if rolePriorityMap[pod.Labels[constant.RoleLabelKey]] < leaderPriority {
- followerCount++
- }
- }
- end := followerCount / 2
- for i := 0; i < end; i++ {
- nextStep := &Step{}
- nextStep.Obj = podList[i]
- start.NextSteps = append(start.NextSteps, nextStep)
- }
-
- if len(start.NextSteps) > 0 {
- start = start.NextSteps[0]
- }
- // append the other 1/2 followers
- podList = podList[end:]
- end = followerCount - end
- for i := 0; i < end; i++ {
- nextStep := &Step{}
- nextStep.Obj = podList[i]
- start.NextSteps = append(start.NextSteps, nextStep)
- }
-
- if len(start.NextSteps) > 0 {
- start = start.NextSteps[0]
- }
- // append leader
- podList = podList[end:]
- for _, pod := range podList {
- nextStep := &Step{}
- nextStep.Obj = pod
- start.NextSteps = append(start.NextSteps, nextStep)
- }
-}
-
-// unknown & empty & leader & followers & learner
-func generateConsensusParallelPlan(plan *Plan, pods []corev1.Pod, rolePriorityMap map[string]int) {
- start := plan.Start
- for _, pod := range pods {
- nextStep := &Step{}
- nextStep.Obj = pod
- start.NextSteps = append(start.NextSteps, nextStep)
- }
-}
-
-// unknown -> empty -> learner -> followers(none->readonly->readwrite) -> leader
-func generateConsensusSerialPlan(plan *Plan, pods []corev1.Pod, rolePriorityMap map[string]int) {
- start := plan.Start
- for _, pod := range pods {
- nextStep := &Step{}
- nextStep.Obj = pod
- start.NextSteps = append(start.NextSteps, nextStep)
- start = nextStep
- }
-}
-
-// ComposeRolePriorityMap generates a priority map based on roles.
-func ComposeRolePriorityMap(consensusSpec *appsv1alpha1.ConsensusSetSpec) map[string]int {
- if consensusSpec == nil {
- consensusSpec = appsv1alpha1.NewConsensusSetSpec()
- }
- rolePriorityMap := make(map[string]int, 0)
- rolePriorityMap[""] = emptyConsensusPriority
- rolePriorityMap[consensusSpec.Leader.Name] = leaderPriority
- if consensusSpec.Learner != nil {
- rolePriorityMap[consensusSpec.Learner.Name] = learnerPriority
- }
- for _, follower := range consensusSpec.Followers {
- switch follower.AccessMode {
- case appsv1alpha1.None:
- rolePriorityMap[follower.Name] = followerNonePriority
- case appsv1alpha1.Readonly:
- rolePriorityMap[follower.Name] = followerReadonlyPriority
- case appsv1alpha1.ReadWrite:
- rolePriorityMap[follower.Name] = followerReadWritePriority
- }
- }
- return rolePriorityMap
-}
-
-// UpdateConsensusSetRoleLabel updates pod role label when internal container role changed
-func UpdateConsensusSetRoleLabel(cli client.Client,
- reqCtx intctrlutil.RequestCtx,
- event *corev1.Event,
- componentDef *appsv1alpha1.ClusterComponentDefinition,
- pod *corev1.Pod, role string) error {
- if componentDef == nil {
- return nil
- }
- return updateConsensusSetRoleLabel(cli, reqCtx, event, componentDef.ConsensusSpec, pod, role)
-}
-
-func updateConsensusSetRoleLabel(cli client.Client,
- reqCtx intctrlutil.RequestCtx,
- event *corev1.Event,
- consensusSpec *appsv1alpha1.ConsensusSetSpec,
- pod *corev1.Pod, role string) error {
- ctx := reqCtx.Ctx
- roleMap := composeConsensusRoleMap(consensusSpec)
- // role not defined in CR, ignore it
- if _, ok := roleMap[role]; !ok {
- return nil
- }
-
- // update pod role label
- patch := client.MergeFrom(pod.DeepCopy())
- pod.Labels[constant.RoleLabelKey] = role
- pod.Labels[constant.ConsensusSetAccessModeLabelKey] = string(roleMap[role].accessMode)
- if pod.Annotations == nil {
- pod.Annotations = map[string]string{}
- }
- pod.Annotations[constant.LastRoleChangedEventTimestampAnnotationKey] = event.EventTime.Time.Format(time.RFC3339Nano)
- return cli.Patch(ctx, pod, patch)
-}
-
-func putConsensusMemberExt(roleMap map[string]consensusMemberExt, name string, role consensusRole, accessMode appsv1alpha1.AccessMode) {
- if roleMap == nil {
- return
- }
-
- if name == "" || role == "" || accessMode == "" {
- return
- }
-
- memberExt := consensusMemberExt{
- name: name,
- consensusRole: role,
- accessMode: accessMode,
- }
-
- roleMap[name] = memberExt
-}
-
-func composeConsensusRoleMap(consensusSpec *appsv1alpha1.ConsensusSetSpec) map[string]consensusMemberExt {
- roleMap := make(map[string]consensusMemberExt, 0)
- putConsensusMemberExt(roleMap,
- consensusSpec.Leader.Name,
- roleLeader,
- consensusSpec.Leader.AccessMode)
-
- for _, follower := range consensusSpec.Followers {
- putConsensusMemberExt(roleMap,
- follower.Name,
- roleFollower,
- follower.AccessMode)
- }
-
- if consensusSpec.Learner != nil {
- putConsensusMemberExt(roleMap,
- consensusSpec.Learner.Name,
- roleLearner,
- consensusSpec.Learner.AccessMode)
- }
-
- return roleMap
-}
-
-func setConsensusSetStatusLeader(consensusSetStatus *appsv1alpha1.ConsensusSetStatus, memberExt consensusMemberExt) bool {
- if consensusSetStatus.Leader.Pod == memberExt.podName {
- return false
- }
- consensusSetStatus.Leader.Pod = memberExt.podName
- consensusSetStatus.Leader.AccessMode = memberExt.accessMode
- consensusSetStatus.Leader.Name = memberExt.name
- return true
-}
-
-func setConsensusSetStatusFollower(consensusSetStatus *appsv1alpha1.ConsensusSetStatus, memberExt consensusMemberExt) bool {
- for _, member := range consensusSetStatus.Followers {
- if member.Pod == memberExt.podName {
- return false
- }
- }
- member := appsv1alpha1.ConsensusMemberStatus{
- Pod: memberExt.podName,
- AccessMode: memberExt.accessMode,
- Name: memberExt.name,
- }
- consensusSetStatus.Followers = append(consensusSetStatus.Followers, member)
- sort.SliceStable(consensusSetStatus.Followers, func(i, j int) bool {
- fi := consensusSetStatus.Followers[i]
- fj := consensusSetStatus.Followers[j]
- return strings.Compare(fi.Pod, fj.Pod) < 0
- })
- return true
-}
-
-func setConsensusSetStatusLearner(consensusSetStatus *appsv1alpha1.ConsensusSetStatus, memberExt consensusMemberExt) bool {
- if consensusSetStatus.Learner == nil {
- consensusSetStatus.Learner = &appsv1alpha1.ConsensusMemberStatus{}
- }
- if consensusSetStatus.Learner.Pod == memberExt.podName {
- return false
- }
- consensusSetStatus.Learner.Pod = memberExt.podName
- consensusSetStatus.Learner.AccessMode = memberExt.accessMode
- consensusSetStatus.Learner.Name = memberExt.name
- return true
-}
-
-func resetConsensusSetStatusRole(consensusSetStatus *appsv1alpha1.ConsensusSetStatus, podName string) {
- // reset leader
- if consensusSetStatus.Leader.Pod == podName {
- consensusSetStatus.Leader.Pod = constant.ComponentStatusDefaultPodName
- consensusSetStatus.Leader.AccessMode = appsv1alpha1.None
- consensusSetStatus.Leader.Name = ""
- }
-
- // reset follower
- for index, member := range consensusSetStatus.Followers {
- if member.Pod == podName {
- consensusSetStatus.Followers = append(consensusSetStatus.Followers[:index], consensusSetStatus.Followers[index+1:]...)
- }
- }
-
- // reset learner
- if consensusSetStatus.Learner != nil && consensusSetStatus.Learner.Pod == podName {
- consensusSetStatus.Learner = nil
- }
-}
-
-func setConsensusSetStatusRoles(
- consensusSetStatus *appsv1alpha1.ConsensusSetStatus,
- consensusSpec *appsv1alpha1.ConsensusSetSpec,
- pods []corev1.Pod) {
- for _, pod := range pods {
- if !intctrlutil.PodIsReadyWithLabel(pod) {
- continue
- }
-
- role := pod.Labels[constant.RoleLabelKey]
- _ = setConsensusSetStatusRole(consensusSetStatus, consensusSpec, role, pod.Name)
- }
-}
-
-func setConsensusSetStatusRole(
- consensusSetStatus *appsv1alpha1.ConsensusSetStatus,
- consensusSpec *appsv1alpha1.ConsensusSetSpec,
- role, podName string) bool {
- // mapping role label to consensus member
- roleMap := composeConsensusRoleMap(consensusSpec)
- memberExt, ok := roleMap[role]
- if !ok {
- return false
- }
- memberExt.podName = podName
- resetConsensusSetStatusRole(consensusSetStatus, memberExt.podName)
- // update cluster.status
- needUpdate := false
- switch memberExt.consensusRole {
- case roleLeader:
- needUpdate = setConsensusSetStatusLeader(consensusSetStatus, memberExt)
- case roleFollower:
- needUpdate = setConsensusSetStatusFollower(consensusSetStatus, memberExt)
- case roleLearner:
- needUpdate = setConsensusSetStatusLearner(consensusSetStatus, memberExt)
- }
- return needUpdate
-}
-
-func updateConsensusRoleInfo(ctx context.Context,
- cli client.Client,
- cluster *appsv1alpha1.Cluster,
- consensusSpec *appsv1alpha1.ConsensusSetSpec,
- componentName string,
- compDefName string,
- pods []corev1.Pod) ([]graph.Vertex, error) {
- leader, followers := composeRoleEnv(consensusSpec, pods)
- ml := client.MatchingLabels{
- constant.AppInstanceLabelKey: cluster.GetName(),
- constant.KBAppComponentLabelKey: componentName,
- constant.AppConfigTypeLabelKey: "kubeblocks-env",
- }
- configList := &corev1.ConfigMapList{}
- if err := cli.List(ctx, configList, client.InNamespace(cluster.Namespace), ml); err != nil {
- return nil, err
- }
-
- vertexes := make([]graph.Vertex, 0)
- for idx := range configList.Items {
- config := configList.Items[idx]
- if config.Data == nil {
- config.Data = make(map[string]string)
- }
-
- config.Data["KB_LEADER"] = leader
- config.Data["KB_FOLLOWERS"] = followers
- // TODO: need to deprecate 'compDefName' being part of variable name, as it's redundant
- // and introduce env/cm key naming reference complexity
- config.Data["KB_"+strings.ToUpper(compDefName)+"_LEADER"] = leader
- config.Data["KB_"+strings.ToUpper(compDefName)+"_FOLLOWERS"] = followers
- vertexes = append(vertexes, &ictrltypes.LifecycleVertex{
- Obj: &config,
- Action: ictrltypes.ActionUpdatePtr(),
- })
- }
-
- // patch pods' annotations
- for idx := range pods {
- pod := pods[idx]
- if pod.Annotations == nil {
- pod.Annotations = map[string]string{}
- }
- pod.Annotations[constant.LeaderAnnotationKey] = leader
- vertexes = append(vertexes, &ictrltypes.LifecycleVertex{
- Obj: &pod,
- Action: ictrltypes.ActionUpdatePtr(),
- })
- }
-
- return vertexes, nil
-}
-
-func composeRoleEnv(consensusSpec *appsv1alpha1.ConsensusSetSpec, pods []corev1.Pod) (leader, followers string) {
- leader, followers = "", ""
- for _, pod := range pods {
- if !intctrlutil.PodIsReadyWithLabel(pod) {
- continue
- }
- role := pod.Labels[constant.RoleLabelKey]
- // mapping role label to consensus member
- roleMap := composeConsensusRoleMap(consensusSpec)
- memberExt, ok := roleMap[role]
- if !ok {
- continue
- }
- switch memberExt.consensusRole {
- case roleLeader:
- leader = pod.Name
- case roleFollower:
- if len(followers) > 0 {
- followers += ","
- }
- followers += pod.Name
- case roleLearner:
- // TODO: CT
- }
- }
- return
-}
diff --git a/controllers/apps/components/consensus_set_utils_test.go b/controllers/apps/components/consensus_set_utils_test.go
deleted file mode 100644
index 88c1cb8a38c..00000000000
--- a/controllers/apps/components/consensus_set_utils_test.go
+++ /dev/null
@@ -1,208 +0,0 @@
-/*
-Copyright (C) 2022-2023 ApeCloud Co., Ltd
-
-This file is part of KubeBlocks project
-
-This program is free software: you can redistribute it and/or modify
-it under the terms of the GNU Affero General Public License as published by
-the Free Software Foundation, either version 3 of the License, or
-(at your option) any later version.
-
-This program is distributed in the hope that it will be useful
-but WITHOUT ANY WARRANTY; without even the implied warranty of
-MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
-GNU Affero General Public License for more details.
-
-You should have received a copy of the GNU Affero General Public License
-along with this program. If not, see .
-*/
-
-package components
-
-import (
- "strconv"
- "testing"
- "time"
-
- "github.com/stretchr/testify/assert"
- apps "k8s.io/api/apps/v1"
- v1 "k8s.io/api/core/v1"
- metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
- "k8s.io/apimachinery/pkg/util/rand"
-
- appsv1alpha1 "github.com/apecloud/kubeblocks/apis/apps/v1alpha1"
- "github.com/apecloud/kubeblocks/internal/constant"
- "github.com/apecloud/kubeblocks/internal/controllerutil"
- testk8s "github.com/apecloud/kubeblocks/internal/testutil/k8s"
-)
-
-func TestIsReady(t *testing.T) {
- set := testk8s.NewFakeStatefulSet("foo", 3)
- pod := testk8s.NewFakeStatefulSetPod(set, 1)
- pod.Status.Conditions = []v1.PodCondition{
- {
- Type: v1.PodReady,
- Status: v1.ConditionTrue,
- },
- }
- pod.Labels = map[string]string{constant.RoleLabelKey: "leader"}
- if !controllerutil.PodIsReadyWithLabel(*pod) {
- t.Errorf("isReady returned false negative")
- }
-}
-
-func TestInitClusterComponentStatusIfNeed(t *testing.T) {
- componentName := "foo"
- cluster := &appsv1alpha1.Cluster{
- Spec: appsv1alpha1.ClusterSpec{
- ComponentSpecs: []appsv1alpha1.ClusterComponentSpec{
- {
- Name: componentName,
- ComponentDefRef: componentName,
- },
- },
- },
- }
- if err := initClusterComponentStatusIfNeed(cluster, componentName, appsv1alpha1.Consensus); err != nil {
- t.Errorf("caught error %v", err)
- }
-
- if len(cluster.Status.Components) == 0 {
- t.Errorf("cluster.Status.ComponentDefs[*] not initialized properly")
- }
- if _, ok := cluster.Status.Components[componentName]; !ok {
- t.Errorf("cluster.Status.ComponentDefs[componentName] not initialized properly")
- }
- consensusSetStatus := cluster.Status.Components[componentName].ConsensusSetStatus
- if consensusSetStatus == nil {
- t.Errorf("cluster.Status.ComponentDefs[componentName].ConsensusSetStatus not initialized properly")
- } else if consensusSetStatus.Leader.Name != "" ||
- consensusSetStatus.Leader.AccessMode != appsv1alpha1.None ||
- consensusSetStatus.Leader.Pod != constant.ComponentStatusDefaultPodName {
- t.Errorf("cluster.Status.ComponentDefs[componentName].ConsensusSetStatus.Leader not initialized properly")
- }
-}
-
-func TestGetPodRevision(t *testing.T) {
- set := testk8s.NewFakeStatefulSet("foo", 3)
- pod := testk8s.NewFakeStatefulSetPod(set, 1)
- if controllerutil.GetPodRevision(pod) != "" {
- t.Errorf("revision should be empty")
- }
-
- pod.Labels = make(map[string]string, 0)
- pod.Labels[apps.StatefulSetRevisionLabel] = "bar"
-
- if controllerutil.GetPodRevision(pod) != "bar" {
- t.Errorf("revision not matched")
- }
-}
-
-func TestSortPods(t *testing.T) {
- createMockPods := func(replicas int, stsName string) []v1.Pod {
- pods := make([]v1.Pod, replicas)
- for i := 0; i < replicas; i++ {
- pods[i] = v1.Pod{
- ObjectMeta: metav1.ObjectMeta{
- Name: stsName + "-" + strconv.Itoa(i),
- Namespace: "default",
- Labels: map[string]string{
- constant.RoleLabelKey: "learner",
- },
- },
- }
- }
- return pods
- }
- randSort := func(pods []v1.Pod) []v1.Pod {
- n := len(pods)
- newPod := make([]v1.Pod, n)
- copy(newPod, pods)
- for i := n; i > 0; i-- {
- randIndex := rand.Intn(i)
- newPod[n-1], newPod[randIndex] = newPod[randIndex], newPod[n-1]
- }
- return newPod
- }
-
- type args struct {
- pods []v1.Pod
- rolePriorityMap map[string]int
- }
- tests := []struct {
- name string
- args args
- want []v1.Pod
- wantErr bool
- }{{
- name: "test_normal",
- args: args{
- rolePriorityMap: map[string]int{
- "learner": 10,
- },
- },
- want: createMockPods(8, "for-test"),
- wantErr: false,
- }, {
- name: "badcase",
- args: args{
- rolePriorityMap: map[string]int{
- "learner": 10,
- },
- },
- want: createMockPods(12, "for-test"),
- wantErr: false,
- }}
- for _, tt := range tests {
- t.Run(tt.name, func(t *testing.T) {
- tt.args.pods = randSort(tt.want)
- SortPods(tt.args.pods, tt.args.rolePriorityMap, constant.RoleLabelKey)
- if !tt.wantErr {
- assert.Equal(t, tt.args.pods, tt.want)
- }
- })
- }
-}
-
-func TestComposeRoleEnv(t *testing.T) {
- componentDef := &appsv1alpha1.ClusterComponentDefinition{
- WorkloadType: appsv1alpha1.Consensus,
- ConsensusSpec: &appsv1alpha1.ConsensusSetSpec{
- Leader: appsv1alpha1.ConsensusMember{
- Name: "leader",
- AccessMode: appsv1alpha1.ReadWrite,
- },
- Followers: []appsv1alpha1.ConsensusMember{
- {
- Name: "follower",
- AccessMode: appsv1alpha1.Readonly,
- },
- },
- },
- }
-
- set := testk8s.NewFakeStatefulSet("foo", 3)
- pods := make([]v1.Pod, 0)
- for i := 0; i < 5; i++ {
- pod := testk8s.NewFakeStatefulSetPod(set, i)
- pod.Status.Conditions = []v1.PodCondition{
- {
- Type: v1.PodReady,
- Status: v1.ConditionTrue,
- },
- }
- pod.Labels = map[string]string{constant.RoleLabelKey: "follower"}
- pods = append(pods, *pod)
- }
- pods[0].Labels = map[string]string{constant.RoleLabelKey: "leader"}
- leader, followers := composeRoleEnv(componentDef.ConsensusSpec, pods)
- assert.Equal(t, "foo-0", leader)
- assert.Equal(t, "foo-1,foo-2,foo-3,foo-4", followers)
-
- dt := time.Now()
- pods[3].DeletionTimestamp = &metav1.Time{Time: dt}
- pods[4].DeletionTimestamp = &metav1.Time{Time: dt}
- leader, followers = composeRoleEnv(componentDef.ConsensusSpec, pods)
- assert.Equal(t, "foo-0", leader)
- assert.Equal(t, "foo-1,foo-2", followers)
-}
diff --git a/controllers/apps/components/consensus_workload.go b/controllers/apps/components/consensus_workload.go
deleted file mode 100644
index 95e9a383c11..00000000000
--- a/controllers/apps/components/consensus_workload.go
+++ /dev/null
@@ -1,56 +0,0 @@
-/*
-Copyright (C) 2022-2023 ApeCloud Co., Ltd
-
-This file is part of KubeBlocks project
-
-This program is free software: you can redistribute it and/or modify
-it under the terms of the GNU Affero General Public License as published by
-the Free Software Foundation, either version 3 of the License, or
-(at your option) any later version.
-
-This program is distributed in the hope that it will be useful
-but WITHOUT ANY WARRANTY; without even the implied warranty of
-MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
-GNU Affero General Public License for more details.
-
-You should have received a copy of the GNU Affero General Public License
-along with this program. If not, see .
-*/
-
-package components
-
-import (
- "github.com/apecloud/kubeblocks/internal/controller/factory"
- "sigs.k8s.io/controller-runtime/pkg/client"
-
- "github.com/apecloud/kubeblocks/internal/constant"
-)
-
-type consensusComponentWorkloadBuilder struct {
- componentWorkloadBuilderBase
-}
-
-var _ componentWorkloadBuilder = &consensusComponentWorkloadBuilder{}
-
-func (b *consensusComponentWorkloadBuilder) BuildWorkload() componentWorkloadBuilder {
- return b.BuildWorkload4StatefulSet("consensus")
-}
-
-func (b *consensusComponentWorkloadBuilder) BuildService() componentWorkloadBuilder {
- buildfn := func() ([]client.Object, error) {
- svcList, err := factory.BuildSvcList(b.Comp.GetCluster(), b.Comp.GetSynthesizedComponent())
- if err != nil {
- return nil, err
- }
- objs := make([]client.Object, 0, len(svcList))
- leader := b.Comp.GetConsensusSpec().Leader
- for _, svc := range svcList {
- if len(leader.Name) > 0 {
- svc.Spec.Selector[constant.RoleLabelKey] = leader.Name
- }
- objs = append(objs, svc)
- }
- return objs, err
- }
- return b.BuildWrapper(buildfn)
-}
diff --git a/controllers/apps/components/base_stateful_hscale.go b/controllers/apps/components/hscale_volume_populator.go
similarity index 50%
rename from controllers/apps/components/base_stateful_hscale.go
rename to controllers/apps/components/hscale_volume_populator.go
index 0dca68793fd..c1f72256424 100644
--- a/controllers/apps/components/base_stateful_hscale.go
+++ b/controllers/apps/components/hscale_volume_populator.go
@@ -20,24 +20,24 @@ along with this program. If not, see .
package components
import (
+ "context"
"fmt"
snapshotv1 "github.com/kubernetes-csi/external-snapshotter/client/v6/apis/volumesnapshot/v1"
appsv1 "k8s.io/api/apps/v1"
- v1 "k8s.io/api/batch/v1"
corev1 "k8s.io/api/core/v1"
+ storagev1 "k8s.io/api/storage/v1"
"k8s.io/apimachinery/pkg/api/errors"
"k8s.io/apimachinery/pkg/types"
"sigs.k8s.io/controller-runtime/pkg/client"
appsv1alpha1 "github.com/apecloud/kubeblocks/apis/apps/v1alpha1"
- dataprotectionv1alpha1 "github.com/apecloud/kubeblocks/apis/dataprotection/v1alpha1"
+ dpv1alpha1 "github.com/apecloud/kubeblocks/apis/dataprotection/v1alpha1"
"github.com/apecloud/kubeblocks/internal/constant"
"github.com/apecloud/kubeblocks/internal/controller/component"
"github.com/apecloud/kubeblocks/internal/controller/factory"
"github.com/apecloud/kubeblocks/internal/controller/plan"
intctrlutil "github.com/apecloud/kubeblocks/internal/controllerutil"
- viper "github.com/apecloud/kubeblocks/internal/viperx"
)
type dataClone interface {
@@ -50,9 +50,8 @@ type dataClone interface {
checkBackupStatus() (backupStatus, error)
backup() ([]client.Object, error)
- pvcKeysToRestore() []types.NamespacedName
- checkRestoreStatus(types.NamespacedName) (backupStatus, error)
- restore(name types.NamespacedName) ([]client.Object, error)
+ checkRestoreStatus(startingIndex int32) (backupStatus, error)
+ restore(startingIndex int32) ([]client.Object, error)
}
type backupStatus string
@@ -88,19 +87,6 @@ func newDataClone(reqCtx intctrlutil.RequestCtx,
}, nil
}
if component.HorizontalScalePolicy.Type == appsv1alpha1.HScaleDataClonePolicyCloneVolume {
- if viper.GetBool("VOLUMESNAPSHOT") {
- return &snapshotDataClone{
- baseDataClone{
- reqCtx: reqCtx,
- cli: cli,
- cluster: cluster,
- component: component,
- stsObj: stsObj,
- stsProto: stsProto,
- key: key,
- },
- }, nil
- }
return &backupDataClone{
baseDataClone{
reqCtx: reqCtx,
@@ -153,17 +139,15 @@ func (d *baseDataClone) cloneData(realDataClone dataClone) ([]client.Object, err
panic(fmt.Sprintf("unexpected backup status: %s, clustre: %s, component: %s",
status, d.cluster.Name, d.component.Name))
}
-
// backup's ready, then start to check restore
- for _, pvcKey := range d.pvcKeysToRestore() {
- restoreStatus, err := realDataClone.checkRestoreStatus(pvcKey)
+ for i := *d.stsObj.Spec.Replicas; i < d.component.Replicas; i++ {
+ restoreStatus, err := realDataClone.checkRestoreStatus(i)
if err != nil {
return nil, err
}
switch restoreStatus {
case backupStatusNotCreated:
-
- restoreObjs, err := realDataClone.restore(pvcKey)
+ restoreObjs, err := realDataClone.restore(i)
if err != nil {
return nil, err
}
@@ -176,7 +160,6 @@ func (d *baseDataClone) cloneData(realDataClone dataClone) ([]client.Object, err
status, d.cluster.Name, d.component.Name))
}
}
-
// create PVCs that do not need to restore
pvcObjs, err := d.createPVCs(d.excludeBackupVCTs())
if err != nil {
@@ -224,17 +207,7 @@ func (d *baseDataClone) allVCTs() []*corev1.PersistentVolumeClaimTemplate {
}
func (d *baseDataClone) backupVCT() *corev1.PersistentVolumeClaimTemplate {
- // TODO: is it possible that component.VolumeClaimTemplates may be empty?
- vct := d.component.VolumeClaimTemplates[0]
- for _, tmpVct := range d.component.VolumeClaimTemplates {
- for _, volumeType := range d.component.VolumeTypes {
- if volumeType.Type == appsv1alpha1.VolumeTypeData && volumeType.Name == tmpVct.Name {
- vct = tmpVct
- break
- }
- }
- }
- return &vct
+ return backupVCT(d.component)
}
func (d *baseDataClone) excludeBackupVCTs() []*corev1.PersistentVolumeClaimTemplate {
@@ -249,19 +222,6 @@ func (d *baseDataClone) excludeBackupVCTs() []*corev1.PersistentVolumeClaimTempl
return vcts
}
-func (d *baseDataClone) pvcKeysToRestore() []types.NamespacedName {
- var pvcKeys []types.NamespacedName
- backupVct := d.backupVCT()
- for i := *d.stsObj.Spec.Replicas; i < d.component.Replicas; i++ {
- pvcKey := types.NamespacedName{
- Namespace: d.stsObj.Namespace,
- Name: fmt.Sprintf("%s-%s-%d", backupVct.Name, d.stsObj.Name, i),
- }
- pvcKeys = append(pvcKeys, pvcKey)
- }
- return pvcKeys
-}
-
func (d *baseDataClone) createPVCs(vcts []*corev1.PersistentVolumeClaimTemplate) ([]client.Object, error) {
objs := make([]client.Object, 0)
for i := *d.stsObj.Spec.Replicas; i < d.component.Replicas; i++ {
@@ -275,18 +235,15 @@ func (d *baseDataClone) createPVCs(vcts []*corev1.PersistentVolumeClaimTemplate)
} else if exist {
continue
}
- pvc, err := factory.BuildPVC(d.cluster, d.component, vct, pvcKey, "")
- if err != nil {
- return nil, err
- }
+ pvc := factory.BuildPVC(d.cluster, d.component, vct, pvcKey, "")
objs = append(objs, pvc)
}
}
return objs, nil
}
-func (d *baseDataClone) getBackupMatchingLabels() client.MatchingLabels {
- return client.MatchingLabels{
+func (d *baseDataClone) getBRLabels() map[string]string {
+ return map[string]string{
constant.AppInstanceLabelKey: d.cluster.Name,
constant.KBAppComponentLabelKey: d.component.Name,
constant.KBManagedByKey: "cluster", // the resources are managed by which controller
@@ -319,223 +276,14 @@ func (d *dummyDataClone) backup() ([]client.Object, error) {
panic("runtime error: dummyDataClone.backup called")
}
-func (d *dummyDataClone) checkRestoreStatus(types.NamespacedName) (backupStatus, error) {
+func (d *dummyDataClone) checkRestoreStatus(startingIndex int32) (backupStatus, error) {
return backupStatusReadyToUse, nil
}
-func (d *dummyDataClone) restore(name types.NamespacedName) ([]client.Object, error) {
+func (d *dummyDataClone) restore(startingIndex int32) ([]client.Object, error) {
panic("runtime error: dummyDataClone.restore called")
}
-type snapshotDataClone struct {
- baseDataClone
-}
-
-var _ dataClone = &snapshotDataClone{}
-
-func (d *snapshotDataClone) succeed() (bool, error) {
- if len(d.component.VolumeClaimTemplates) == 0 {
- d.reqCtx.Recorder.Eventf(d.cluster,
- corev1.EventTypeNormal,
- "HorizontalScale",
- "no VolumeClaimTemplates, no need to do data clone.")
- return true, nil
- }
- return d.checkAllPVCsExist()
-}
-
-func (d *snapshotDataClone) clearTmpResources() ([]client.Object, error) {
- allPVCBound, err := d.isAllPVCBound()
- if err != nil {
- return nil, err
- }
- if !allPVCBound {
- return nil, nil
- }
- return d.deleteSnapshot()
-}
-
-func (d *snapshotDataClone) backup() ([]client.Object, error) {
- objs := make([]client.Object, 0)
- backupPolicyTplName := d.component.HorizontalScalePolicy.BackupPolicyTemplateName
-
- backupPolicyTemplate := &appsv1alpha1.BackupPolicyTemplate{}
- err := d.cli.Get(d.reqCtx.Ctx, client.ObjectKey{Name: backupPolicyTplName}, backupPolicyTemplate)
- if err != nil {
- return nil, err
- }
-
- // if there is backuppolicytemplate created by provider
- backupPolicy, err := getBackupPolicyFromTemplate(d.reqCtx, d.cli, d.cluster, d.component.ComponentDef, backupPolicyTplName)
- if err != nil {
- return nil, err
- }
- if backupPolicy == nil {
- return nil, intctrlutil.NewNotFound("not found any backup policy created by %s", backupPolicyTplName)
- }
- backup, err := factory.BuildBackup(d.cluster, d.component, backupPolicy.Name, d.key, "snapshot")
- if err != nil {
- return nil, err
- }
- objs = append(objs, backup)
- d.reqCtx.Recorder.Eventf(d.cluster, corev1.EventTypeNormal, "BackupJobCreate", "Create backupJob/%s", d.key.Name)
- return objs, nil
-}
-
-func (d *snapshotDataClone) checkBackupStatus() (backupStatus, error) {
- backupPolicyTplName := d.component.HorizontalScalePolicy.BackupPolicyTemplateName
- backupPolicyTemplate := &appsv1alpha1.BackupPolicyTemplate{}
- err := d.cli.Get(d.reqCtx.Ctx, client.ObjectKey{Name: backupPolicyTplName}, backupPolicyTemplate)
- if err != nil {
- return backupStatusFailed, err
- }
- backup := dataprotectionv1alpha1.Backup{}
- if err := d.cli.Get(d.reqCtx.Ctx, d.key, &backup); err != nil {
- if errors.IsNotFound(err) {
- return backupStatusNotCreated, nil
- } else {
- return backupStatusFailed, err
- }
- }
- if backup.Status.Phase == dataprotectionv1alpha1.BackupFailed {
- return backupStatusFailed, intctrlutil.NewErrorf(intctrlutil.ErrorTypeBackupFailed, "backup for horizontalScaling failed: %s",
- backup.Status.FailureReason)
- }
- if backup.Status.Phase != dataprotectionv1alpha1.BackupCompleted {
- return backupStatusProcessing, nil
- }
- return backupStatusReadyToUse, nil
-}
-
-func (d *snapshotDataClone) restore(pvcKey types.NamespacedName) ([]client.Object, error) {
- objs := make([]client.Object, 0)
- vct := d.backupVCT()
- // create pvc from snapshot for every new pod
- if pvc, err := d.checkedCreatePVCFromSnapshot(
- pvcKey,
- vct); err != nil {
- d.reqCtx.Log.Error(err, "checkedCreatePVCFromSnapshot failed")
- return nil, err
- } else if pvc != nil {
- objs = append(objs, pvc)
- }
- return objs, nil
-}
-
-func (d *snapshotDataClone) checkRestoreStatus(pvcKey types.NamespacedName) (backupStatus, error) {
- pvc := corev1.PersistentVolumeClaim{}
- if err := d.cli.Get(d.reqCtx.Ctx, pvcKey, &pvc); err != nil {
- if errors.IsNotFound(err) {
- return backupStatusNotCreated, nil
- }
- return backupStatusFailed, err
- }
- return backupStatusReadyToUse, nil
-}
-
-func (d *snapshotDataClone) listVolumeSnapshotByLabels(vsList *snapshotv1.VolumeSnapshotList, ml client.MatchingLabels) error {
- compatClient := intctrlutil.VolumeSnapshotCompatClient{ReadonlyClient: d.cli, Ctx: d.reqCtx.Ctx}
- // get vs from backup.
- backupList := dataprotectionv1alpha1.BackupList{}
- if err := d.cli.List(d.reqCtx.Ctx, &backupList, client.InNamespace(d.cluster.Namespace), ml); err != nil {
- return err
- } else if len(backupList.Items) == 0 {
- // ignore not found
- return nil
- }
- return compatClient.List(vsList, client.MatchingLabels{
- constant.DataProtectionLabelBackupNameKey: backupList.Items[0].Name,
- })
-}
-
-func (d *snapshotDataClone) checkedCreatePVCFromSnapshot(pvcKey types.NamespacedName,
- vct *corev1.PersistentVolumeClaimTemplate) (client.Object, error) {
- pvc := corev1.PersistentVolumeClaim{}
- // check pvc existence
- if err := d.cli.Get(d.reqCtx.Ctx, pvcKey, &pvc); err != nil {
- if !errors.IsNotFound(err) {
- return nil, err
- }
- ml := d.getBackupMatchingLabels()
- vsList := snapshotv1.VolumeSnapshotList{}
- if err = d.listVolumeSnapshotByLabels(&vsList, ml); err != nil {
- return nil, err
- }
- if len(vsList.Items) == 0 {
- return nil, fmt.Errorf("volumesnapshot not found in cluster %s component %s", d.cluster.Name, d.component.Name)
- }
- // exclude volumes that are deleting
- vsName := ""
- for _, vs := range vsList.Items {
- if vs.DeletionTimestamp != nil {
- continue
- }
- vsName = vs.Name
- break
- }
- return d.createPVCFromSnapshot(vct, pvcKey, vsName)
- }
- return nil, nil
-}
-
-func (d *snapshotDataClone) createPVCFromSnapshot(
- vct *corev1.PersistentVolumeClaimTemplate,
- pvcKey types.NamespacedName,
- snapshotName string) (client.Object, error) {
- pvc, err := factory.BuildPVC(d.cluster, d.component, vct, pvcKey, snapshotName)
- if err != nil {
- return nil, err
- }
- return pvc, nil
-}
-
-func (d *snapshotDataClone) deleteSnapshot() ([]client.Object, error) {
- objs, err := d.deleteBackup()
- if err != nil {
- return nil, err
- }
- if len(objs) > 0 {
- d.reqCtx.Recorder.Eventf(d.cluster, corev1.EventTypeNormal, "BackupJobDelete", "Delete backupJob/%s", d.key.Name)
- }
-
- return objs, nil
-}
-
-// deleteBackup will delete all backup related resources created during horizontal scaling
-func (d *snapshotDataClone) deleteBackup() ([]client.Object, error) {
- ml := d.getBackupMatchingLabels()
- backupList := dataprotectionv1alpha1.BackupList{}
- if err := d.cli.List(d.reqCtx.Ctx, &backupList, client.InNamespace(d.cluster.Namespace), ml); err != nil {
- return nil, err
- }
- objs := make([]client.Object, 0)
- for i := range backupList.Items {
- objs = append(objs, &backupList.Items[i])
- }
- return objs, nil
-}
-
-func (d *snapshotDataClone) isAllPVCBound() (bool, error) {
- if len(d.stsObj.Spec.VolumeClaimTemplates) == 0 {
- return true, nil
- }
- for i := 0; i < int(d.component.Replicas); i++ {
- pvcKey := types.NamespacedName{
- Namespace: d.stsObj.Namespace,
- Name: fmt.Sprintf("%s-%s-%d", d.stsObj.Spec.VolumeClaimTemplates[0].Name, d.stsObj.Name, i),
- }
- pvc := corev1.PersistentVolumeClaim{}
- // check pvc existence
- if err := d.cli.Get(d.reqCtx.Ctx, pvcKey, &pvc); err != nil {
- return false, client.IgnoreNotFound(err)
- }
- if pvc.Status.Phase != corev1.ClaimBound {
- return false, nil
- }
- }
- return true, nil
-}
-
type backupDataClone struct {
baseDataClone
}
@@ -554,8 +302,8 @@ func (d *backupDataClone) succeed() (bool, error) {
if err != nil || !allPVCsExist {
return allPVCsExist, err
}
- for _, pvcKey := range d.pvcKeysToRestore() {
- restoreStatus, err := d.checkRestoreStatus(pvcKey)
+ for i := *d.stsObj.Spec.Replicas; i < d.component.Replicas; i++ {
+ restoreStatus, err := d.checkRestoreStatus(i)
if err != nil {
return false, err
}
@@ -569,21 +317,20 @@ func (d *backupDataClone) succeed() (bool, error) {
func (d *backupDataClone) clearTmpResources() ([]client.Object, error) {
objs := make([]client.Object, 0)
// delete backup
- ml := d.getBackupMatchingLabels()
- backupList := dataprotectionv1alpha1.BackupList{}
- if err := d.cli.List(d.reqCtx.Ctx, &backupList, client.InNamespace(d.cluster.Namespace), ml); err != nil {
+ brLabels := d.getBRLabels()
+ backupList := dpv1alpha1.BackupList{}
+ if err := d.cli.List(d.reqCtx.Ctx, &backupList, client.InNamespace(d.cluster.Namespace), client.MatchingLabels(brLabels)); err != nil {
return nil, err
}
for i := range backupList.Items {
objs = append(objs, &backupList.Items[i])
}
- // delete restore job
- jobList := v1.JobList{}
- if err := d.cli.List(d.reqCtx.Ctx, &jobList, client.InNamespace(d.cluster.Namespace), ml); err != nil {
+ restoreList := dpv1alpha1.RestoreList{}
+ if err := d.cli.List(d.reqCtx.Ctx, &restoreList, client.InNamespace(d.cluster.Namespace), client.MatchingLabels(brLabels)); err != nil {
return nil, err
}
- for i := range jobList.Items {
- objs = append(objs, &jobList.Items[i])
+ for i := range restoreList.Items {
+ objs = append(objs, &restoreList.Items[i])
}
return objs, nil
}
@@ -598,16 +345,23 @@ func (d *backupDataClone) backup() ([]client.Object, error) {
if backupPolicy == nil {
return nil, intctrlutil.NewNotFound("not found any backup policy created by %s", backupPolicyTplName)
}
- backup, err := factory.BuildBackup(d.cluster, d.component, backupPolicy.Name, d.key, "datafile")
+ volumeSnapshotEnabled, err := isVolumeSnapshotEnabled(d.reqCtx.Ctx, d.cli, d.stsObj, backupVCT(d.component))
if err != nil {
return nil, err
}
+ backupMethods := getBackupMethods(backupPolicy, volumeSnapshotEnabled)
+ if len(backupMethods) == 0 {
+ return nil, fmt.Errorf("no backup method found in backup policy %s", backupPolicy.Name)
+ } else if len(backupMethods) > 1 {
+ return nil, fmt.Errorf("more than one backup methods found in backup policy %s", backupPolicy.Name)
+ }
+ backup := factory.BuildBackup(d.cluster, d.component, backupPolicy.Name, d.key, backupMethods[0])
objs = append(objs, backup)
return objs, nil
}
func (d *backupDataClone) checkBackupStatus() (backupStatus, error) {
- backup := dataprotectionv1alpha1.Backup{}
+ backup := dpv1alpha1.Backup{}
if err := d.cli.Get(d.reqCtx.Ctx, d.key, &backup); err != nil {
if errors.IsNotFound(err) {
return backupStatusNotCreated, nil
@@ -615,51 +369,37 @@ func (d *backupDataClone) checkBackupStatus() (backupStatus, error) {
return backupStatusFailed, err
}
}
- if backup.Status.Phase == dataprotectionv1alpha1.BackupFailed {
- return backupStatusFailed, fmt.Errorf("failed to backup: %s", backup.Status.FailureReason)
+ if backup.Status.Phase == dpv1alpha1.BackupPhaseFailed {
+ return backupStatusFailed, intctrlutil.NewErrorf(intctrlutil.ErrorTypeBackupFailed, "backup for horizontalScaling failed: %s",
+ backup.Status.FailureReason)
}
- if backup.Status.Phase == dataprotectionv1alpha1.BackupCompleted {
+ if backup.Status.Phase == dpv1alpha1.BackupPhaseCompleted {
return backupStatusReadyToUse, nil
}
return backupStatusProcessing, nil
}
-func (d *backupDataClone) restore(pvcKey types.NamespacedName) ([]client.Object, error) {
- objs := make([]client.Object, 0)
- backup := dataprotectionv1alpha1.Backup{}
- if err := d.cli.Get(d.reqCtx.Ctx, d.key, &backup); err != nil {
- return nil, err
- }
- pvc, err := factory.BuildPVC(d.cluster, d.component, d.backupVCT(), pvcKey, "")
- if err != nil {
- return nil, err
- }
- objs = append(objs, pvc)
- backupTool := &dataprotectionv1alpha1.BackupTool{}
- if err = d.cli.Get(d.reqCtx.Ctx, client.ObjectKey{Name: backup.Status.BackupToolName}, backupTool); err != nil {
+func (d *backupDataClone) restore(startingIndex int32) ([]client.Object, error) {
+ backup := &dpv1alpha1.Backup{}
+ if err := d.cli.Get(d.reqCtx.Ctx, d.key, backup); err != nil {
return nil, err
}
- restoreMgr := plan.NewRestoreManager(d.reqCtx.Ctx, d.cli, d.cluster, nil)
- restoreJobs, err := restoreMgr.BuildDatafileRestoreJobByPVCS(d.baseDataClone.component, &backup, backupTool, []string{pvc.Name}, d.getBackupMatchingLabels())
- if err != nil {
+ restoreMGR := plan.NewRestoreManager(d.reqCtx.Ctx, d.cli, d.cluster, nil, d.getBRLabels(), int32(1), startingIndex)
+ restore, err := restoreMGR.BuildPrepareDataRestore(d.component, backup)
+ if err != nil || restore == nil {
return nil, err
}
- objs = append(objs, restoreJobs...)
- return objs, nil
+ return []client.Object{restore}, nil
}
-func (d *backupDataClone) checkRestoreStatus(pvcKey types.NamespacedName) (backupStatus, error) {
- job := v1.Job{}
- restoreMgr := plan.NewRestoreManager(d.reqCtx.Ctx, d.cli, d.cluster, nil)
- jobName := restoreMgr.GetDatafileRestoreJobName(pvcKey.Name)
- if err := d.cli.Get(d.reqCtx.Ctx, types.NamespacedName{Namespace: pvcKey.Namespace, Name: jobName}, &job); err != nil {
- if errors.IsNotFound(err) {
- return backupStatusNotCreated, nil
- } else {
- return backupStatusNotCreated, err
- }
+func (d *backupDataClone) checkRestoreStatus(startingIndex int32) (backupStatus, error) {
+ restoreMGR := plan.NewRestoreManager(d.reqCtx.Ctx, d.cli, d.cluster, nil, d.getBRLabels(), int32(1), startingIndex)
+ restoreMeta := restoreMGR.GetRestoreObjectMeta(d.component, dpv1alpha1.PrepareData)
+ restore := &dpv1alpha1.Restore{}
+ if err := d.cli.Get(d.reqCtx.Ctx, types.NamespacedName{Namespace: d.cluster.Namespace, Name: restoreMeta.Name}, restore); err != nil {
+ return backupStatusNotCreated, client.IgnoreNotFound(err)
}
- if job.Status.Succeeded == 1 {
+ if restore.Status.Phase == dpv1alpha1.RestorePhaseCompleted {
return backupStatusReadyToUse, nil
}
return backupStatusProcessing, nil
@@ -669,8 +409,8 @@ func (d *backupDataClone) checkRestoreStatus(pvcKey types.NamespacedName) (backu
func getBackupPolicyFromTemplate(reqCtx intctrlutil.RequestCtx,
cli client.Client,
cluster *appsv1alpha1.Cluster,
- componentDef, backupPolicyTemplateName string) (*dataprotectionv1alpha1.BackupPolicy, error) {
- backupPolicyList := &dataprotectionv1alpha1.BackupPolicyList{}
+ componentDef, backupPolicyTemplateName string) (*dpv1alpha1.BackupPolicy, error) {
+ backupPolicyList := &dpv1alpha1.BackupPolicyList{}
if err := cli.List(reqCtx.Ctx, backupPolicyList,
client.InNamespace(cluster.Namespace),
client.MatchingLabels{
@@ -686,3 +426,69 @@ func getBackupPolicyFromTemplate(reqCtx intctrlutil.RequestCtx,
}
return nil, nil
}
+
+func backupVCT(component *component.SynthesizedComponent) *corev1.PersistentVolumeClaimTemplate {
+ if len(component.VolumeClaimTemplates) == 0 {
+ return nil
+ }
+ vct := component.VolumeClaimTemplates[0]
+ for _, tmpVct := range component.VolumeClaimTemplates {
+ for _, volumeType := range component.VolumeTypes {
+ if volumeType.Type == appsv1alpha1.VolumeTypeData && volumeType.Name == tmpVct.Name {
+ vct = tmpVct
+ break
+ }
+ }
+ }
+ return &vct
+}
+
+func isVolumeSnapshotEnabled(ctx context.Context, cli client.Client,
+ sts *appsv1.StatefulSet, vct *corev1.PersistentVolumeClaimTemplate) (bool, error) {
+ if sts == nil || vct == nil {
+ return false, nil
+ }
+ pvcKey := types.NamespacedName{
+ Namespace: sts.Namespace,
+ Name: fmt.Sprintf("%s-%s-%d", vct.Name, sts.Name, 0),
+ }
+ pvc := corev1.PersistentVolumeClaim{}
+ if err := cli.Get(ctx, pvcKey, &pvc); err != nil {
+ return false, client.IgnoreNotFound(err)
+ }
+ if pvc.Spec.StorageClassName == nil {
+ return false, nil
+ }
+
+ storageClass := storagev1.StorageClass{}
+ if err := cli.Get(ctx, types.NamespacedName{Name: *pvc.Spec.StorageClassName}, &storageClass); err != nil {
+ return false, client.IgnoreNotFound(err)
+ }
+
+ vscList := snapshotv1.VolumeSnapshotClassList{}
+ if err := cli.List(ctx, &vscList); err != nil {
+ return false, err
+ }
+ for _, vsc := range vscList.Items {
+ if vsc.Driver == storageClass.Provisioner {
+ return true, nil
+ }
+ }
+ return false, nil
+}
+
+func getBackupMethods(backupPolicy *dpv1alpha1.BackupPolicy, useVolumeSnapshot bool) []string {
+ var vsMethods []string
+ var otherMethods []string
+ for _, method := range backupPolicy.Spec.BackupMethods {
+ if method.SnapshotVolumes != nil && *method.SnapshotVolumes {
+ vsMethods = append(vsMethods, method.Name)
+ } else {
+ otherMethods = append(otherMethods, method.Name)
+ }
+ }
+ if useVolumeSnapshot {
+ return vsMethods
+ }
+ return otherMethods
+}
diff --git a/controllers/apps/components/plan.go b/controllers/apps/components/plan.go
deleted file mode 100644
index f7d5791c2bb..00000000000
--- a/controllers/apps/components/plan.go
+++ /dev/null
@@ -1,82 +0,0 @@
-/*
-Copyright (C) 2022-2023 ApeCloud Co., Ltd
-
-This file is part of KubeBlocks project
-
-This program is free software: you can redistribute it and/or modify
-it under the terms of the GNU Affero General Public License as published by
-the Free Software Foundation, either version 3 of the License, or
-(at your option) any later version.
-
-This program is distributed in the hope that it will be useful
-but WITHOUT ANY WARRANTY; without even the implied warranty of
-MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
-GNU Affero General Public License for more details.
-
-You should have received a copy of the GNU Affero General Public License
-along with this program. If not, see .
-*/
-
-package components
-
-type Plan struct {
- Start *Step
- WalkFunc walkFunc
-}
-
-type Step struct {
- Obj interface{}
- NextSteps []*Step
-}
-
-type walkFunc func(obj interface{}) (bool, error)
-
-// WalkOneStep process plan stepping
-// @return isCompleted
-// @return err
-func (p *Plan) WalkOneStep() (bool, error) {
- if p == nil {
- return true, nil
- }
-
- if len(p.Start.NextSteps) == 0 {
- return true, nil
- }
-
- shouldStop := false
- for _, step := range p.Start.NextSteps {
- walked, err := p.WalkFunc(step.Obj)
- if err != nil {
- return false, err
- }
- if walked {
- shouldStop = true
- }
- }
- if shouldStop {
- return false, nil
- }
-
- // generate new plan
- plan := &Plan{}
- plan.Start = &Step{}
- plan.WalkFunc = p.WalkFunc
- plan.Start.NextSteps = make([]*Step, 0)
- for _, step := range p.Start.NextSteps {
- for _, nextStep := range step.NextSteps {
- if !containStep(plan.Start.NextSteps, nextStep) {
- plan.Start.NextSteps = append(plan.Start.NextSteps, nextStep)
- }
- }
- }
- return plan.WalkOneStep()
-}
-
-func containStep(steps []*Step, step *Step) bool {
- for _, s := range steps {
- if s == step {
- return true
- }
- }
- return false
-}
diff --git a/controllers/apps/components/plan_test.go b/controllers/apps/components/plan_test.go
deleted file mode 100644
index 628064220c0..00000000000
--- a/controllers/apps/components/plan_test.go
+++ /dev/null
@@ -1,67 +0,0 @@
-/*
-Copyright (C) 2022-2023 ApeCloud Co., Ltd
-
-This file is part of KubeBlocks project
-
-This program is free software: you can redistribute it and/or modify
-it under the terms of the GNU Affero General Public License as published by
-the Free Software Foundation, either version 3 of the License, or
-(at your option) any later version.
-
-This program is distributed in the hope that it will be useful
-but WITHOUT ANY WARRANTY; without even the implied warranty of
-MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
-GNU Affero General Public License for more details.
-
-You should have received a copy of the GNU Affero General Public License
-along with this program. If not, see .
-*/
-
-package components
-
-import (
- "testing"
-)
-
-func TestWalkOneStep(t *testing.T) {
- plan := &Plan{}
- plan.Start = &Step{}
- plan.Start.NextSteps = make([]*Step, 1)
- plan.WalkFunc = func(obj interface{}) (bool, error) {
- currentPos := obj.(int)
- if currentPos == 2 {
- return true, nil
- }
-
- return false, nil
- }
-
- step1 := &Step{}
- step1.Obj = 1
- step1.NextSteps = make([]*Step, 1)
- plan.Start.NextSteps[0] = step1
-
- step2 := &Step{}
- step2.Obj = 2
- step1.NextSteps[0] = step2
-
- end, err := plan.WalkOneStep()
- if err != nil {
- t.Errorf("walk error: %v", err)
- }
- if end {
- t.Errorf("walk should not end")
- }
-
- step2.Obj = 3
-
- end, err = plan.WalkOneStep()
-
- if err != nil {
- t.Errorf("walk error: %v", err)
- }
- if !end {
- t.Errorf("walk should end")
- }
-
-}
diff --git a/controllers/apps/components/replication.go b/controllers/apps/components/replication.go
deleted file mode 100644
index 4a98e1cf998..00000000000
--- a/controllers/apps/components/replication.go
+++ /dev/null
@@ -1,107 +0,0 @@
-/*
-Copyright (C) 2022-2023 ApeCloud Co., Ltd
-
-This file is part of KubeBlocks project
-
-This program is free software: you can redistribute it and/or modify
-it under the terms of the GNU Affero General Public License as published by
-the Free Software Foundation, either version 3 of the License, or
-(at your option) any later version.
-
-This program is distributed in the hope that it will be useful
-but WITHOUT ANY WARRANTY; without even the implied warranty of
-MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
-GNU Affero General Public License for more details.
-
-You should have received a copy of the GNU Affero General Public License
-along with this program. If not, see .
-*/
-
-package components
-
-import (
- "k8s.io/client-go/tools/record"
- "sigs.k8s.io/controller-runtime/pkg/client"
-
- appsv1alpha1 "github.com/apecloud/kubeblocks/apis/apps/v1alpha1"
- "github.com/apecloud/kubeblocks/internal/controller/component"
- "github.com/apecloud/kubeblocks/internal/controller/graph"
- ictrltypes "github.com/apecloud/kubeblocks/internal/controller/types"
- intctrlutil "github.com/apecloud/kubeblocks/internal/controllerutil"
-)
-
-func newReplicationComponent(cli client.Client,
- recorder record.EventRecorder,
- cluster *appsv1alpha1.Cluster,
- clusterVersion *appsv1alpha1.ClusterVersion,
- synthesizedComponent *component.SynthesizedComponent,
- dag *graph.DAG) *replicationComponent {
- comp := &replicationComponent{
- statefulComponentBase: statefulComponentBase{
- componentBase: componentBase{
- Client: cli,
- Recorder: recorder,
- Cluster: cluster,
- ClusterVersion: clusterVersion,
- Component: synthesizedComponent,
- ComponentSet: &replicationSet{
- stateful: stateful{
- componentSetBase: componentSetBase{
- Cli: cli,
- Cluster: cluster,
- SynthesizedComponent: synthesizedComponent,
- ComponentSpec: nil,
- ComponentDef: nil,
- },
- },
- },
- Dag: dag,
- WorkloadVertex: nil,
- },
- },
- }
- return comp
-}
-
-type replicationComponent struct {
- statefulComponentBase
-}
-
-var _ Component = &replicationComponent{}
-
-func (c *replicationComponent) newBuilder(reqCtx intctrlutil.RequestCtx, cli client.Client,
- action *ictrltypes.LifecycleAction) componentWorkloadBuilder {
- builder := &replicationComponentWorkloadBuilder{
- componentWorkloadBuilderBase: componentWorkloadBuilderBase{
- ReqCtx: reqCtx,
- Client: cli,
- Comp: c,
- DefaultAction: action,
- Error: nil,
- EnvConfig: nil,
- Workload: nil,
- },
- }
- builder.ConcreteBuilder = builder
- return builder
-}
-
-func (c *replicationComponent) GetWorkloadType() appsv1alpha1.WorkloadType {
- return appsv1alpha1.Replication
-}
-
-func (c *replicationComponent) GetBuiltObjects(reqCtx intctrlutil.RequestCtx, cli client.Client) ([]client.Object, error) {
- return c.statefulComponentBase.GetBuiltObjects(c.newBuilder(reqCtx, cli, ictrltypes.ActionCreatePtr()))
-}
-
-func (c *replicationComponent) Create(reqCtx intctrlutil.RequestCtx, cli client.Client) error {
- return c.statefulComponentBase.Create(reqCtx, cli, c.newBuilder(reqCtx, cli, ictrltypes.ActionCreatePtr()))
-}
-
-func (c *replicationComponent) Update(reqCtx intctrlutil.RequestCtx, cli client.Client) error {
- return c.statefulComponentBase.Update(reqCtx, cli, c.newBuilder(reqCtx, cli, nil))
-}
-
-func (c *replicationComponent) Status(reqCtx intctrlutil.RequestCtx, cli client.Client) error {
- return c.statefulComponentBase.Status(reqCtx, cli, c.newBuilder(reqCtx, cli, ictrltypes.ActionNoopPtr()))
-}
diff --git a/controllers/apps/components/replication_set.go b/controllers/apps/components/replication_set.go
deleted file mode 100644
index b3f922c21de..00000000000
--- a/controllers/apps/components/replication_set.go
+++ /dev/null
@@ -1,277 +0,0 @@
-/*
-Copyright (C) 2022-2023 ApeCloud Co., Ltd
-
-This file is part of KubeBlocks project
-
-This program is free software: you can redistribute it and/or modify
-it under the terms of the GNU Affero General Public License as published by
-the Free Software Foundation, either version 3 of the License, or
-(at your option) any later version.
-
-This program is distributed in the hope that it will be useful
-but WITHOUT ANY WARRANTY; without even the implied warranty of
-MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
-GNU Affero General Public License for more details.
-
-You should have received a copy of the GNU Affero General Public License
-along with this program. If not, see .
-*/
-
-package components
-
-import (
- "context"
- "fmt"
-
- "github.com/pkg/errors"
- appsv1 "k8s.io/api/apps/v1"
- corev1 "k8s.io/api/core/v1"
- "sigs.k8s.io/controller-runtime/pkg/client"
-
- appsv1alpha1 "github.com/apecloud/kubeblocks/apis/apps/v1alpha1"
- "github.com/apecloud/kubeblocks/internal/constant"
- "github.com/apecloud/kubeblocks/internal/controller/graph"
- ictrltypes "github.com/apecloud/kubeblocks/internal/controller/types"
- intctrlutil "github.com/apecloud/kubeblocks/internal/controllerutil"
-)
-
-const (
- emptyReplicationPriority = iota
- secondaryPriority
- primaryPriority
-)
-
-// replicationSet is a component object used by Cluster, ClusterComponentDefinition and ClusterComponentSpec
-type replicationSet struct {
- stateful
-}
-
-var _ componentSet = &replicationSet{}
-
-func (r *replicationSet) getName() string {
- if r.SynthesizedComponent != nil {
- return r.SynthesizedComponent.Name
- }
- return r.ComponentSpec.Name
-}
-
-func (r *replicationSet) getWorkloadType() appsv1alpha1.WorkloadType {
- return appsv1alpha1.Replication
-}
-
-func (r *replicationSet) getReplicas() int32 {
- if r.SynthesizedComponent != nil {
- return r.SynthesizedComponent.Replicas
- }
- return r.ComponentSpec.Replicas
-}
-
-// IsRunning is the implementation of the type Component interface method,
-// which is used to check whether the replicationSet component is running normally.
-func (r *replicationSet) IsRunning(ctx context.Context, obj client.Object) (bool, error) {
- var componentStatusIsRunning = true
- sts := convertToStatefulSet(obj)
- isRevisionConsistent, err := isStsAndPodsRevisionConsistent(ctx, r.Cli, sts)
- if err != nil {
- return false, err
- }
- stsIsReady := statefulSetOfComponentIsReady(sts, isRevisionConsistent, nil)
- if !stsIsReady {
- return false, nil
- }
- if sts.Status.AvailableReplicas < r.getReplicas() {
- componentStatusIsRunning = false
- }
- return componentStatusIsRunning, nil
-}
-
-// PodsReady is the implementation of the type Component interface method,
-// which is used to check whether all the pods of replicationSet component are ready.
-func (r *replicationSet) PodsReady(ctx context.Context, obj client.Object) (bool, error) {
- return r.stateful.PodsReady(ctx, obj)
-}
-
-// PodIsAvailable is the implementation of the type Component interface method,
-// Check whether the status of a Pod of the replicationSet is ready, including the role label on the Pod
-func (r *replicationSet) PodIsAvailable(pod *corev1.Pod, minReadySeconds int32) bool {
- if pod == nil {
- return false
- }
- return intctrlutil.PodIsReadyWithLabel(*pod)
-}
-
-func (r *replicationSet) GetPhaseWhenPodsReadyAndProbeTimeout(pods []*corev1.Pod) (appsv1alpha1.ClusterComponentPhase, appsv1alpha1.ComponentMessageMap) {
- return "", nil
-}
-
-// GetPhaseWhenPodsNotReady is the implementation of the type Component interface method,
-// when the pods of replicationSet are not ready, calculate the component phase is Failed or Abnormal.
-// if return an empty phase, means the pods of component are ready and skips it.
-func (r *replicationSet) GetPhaseWhenPodsNotReady(ctx context.Context,
- componentName string,
- originPhaseIsUpRunning bool) (appsv1alpha1.ClusterComponentPhase, appsv1alpha1.ComponentMessageMap, error) {
- stsList := &appsv1.StatefulSetList{}
- podList, err := getCompRelatedObjectList(ctx, r.Cli, *r.Cluster,
- componentName, stsList)
- if err != nil || len(stsList.Items) == 0 {
- return "", nil, err
- }
- stsObj := stsList.Items[0]
- podCount := len(podList.Items)
- componentReplicas := r.getReplicas()
- if podCount == 0 || stsObj.Status.AvailableReplicas == 0 {
- return getPhaseWithNoAvailableReplicas(componentReplicas), nil, nil
- }
- // get the statefulSet of component
- var (
- existLatestRevisionFailedPod bool
- primaryIsReady bool
- statusMessages = appsv1alpha1.ComponentMessageMap{}
- )
- for _, v := range podList.Items {
- // if the pod is terminating, ignore it
- if v.DeletionTimestamp != nil {
- return "", nil, nil
- }
- labelValue := v.Labels[constant.RoleLabelKey]
- if labelValue == constant.Primary && intctrlutil.PodIsReady(&v) {
- primaryIsReady = true
- continue
- }
- if labelValue == "" {
- statusMessages.SetObjectMessage(v.Kind, v.Name, "empty label for pod, please check.")
- }
- // if component is up running but pod is not ready, this pod should be failed.
- // for example: full disk cause readiness probe failed and serve is not available.
- // but kubelet only sets the container is not ready and pod is also Running.
- if originPhaseIsUpRunning && !intctrlutil.PodIsReady(&v) && intctrlutil.PodIsControlledByLatestRevision(&v, &stsObj) {
- existLatestRevisionFailedPod = true
- continue
- }
- isFailed, _, message := IsPodFailedAndTimedOut(&v)
- if isFailed && intctrlutil.PodIsControlledByLatestRevision(&v, &stsObj) {
- existLatestRevisionFailedPod = true
- statusMessages.SetObjectMessage(v.Kind, v.Name, message)
- }
- }
- return getCompPhaseByConditions(existLatestRevisionFailedPod, primaryIsReady,
- componentReplicas, int32(podCount), stsObj.Status.AvailableReplicas), statusMessages, nil
-}
-
-// HandleRestart is the implementation of the type Component interface method, which is used to handle the restart of the Replication workload.
-func (r *replicationSet) HandleRestart(ctx context.Context, obj client.Object) ([]graph.Vertex, error) {
- if r.getWorkloadType() != appsv1alpha1.Replication {
- return nil, nil
- }
- priorityMapperFn := func(component *appsv1alpha1.ClusterComponentDefinition) map[string]int {
- return composeReplicationRolePriorityMap()
- }
- return r.HandleUpdateWithStrategy(ctx, obj, nil, priorityMapperFn, generateReplicationSerialPlan, generateReplicationBestEffortParallelPlan, generateReplicationParallelPlan)
-}
-
-// HandleRoleChange is the implementation of the type Component interface method, which is used to handle the role change of the Replication workload.
-func (r *replicationSet) HandleRoleChange(ctx context.Context, obj client.Object) ([]graph.Vertex, error) {
- podList, err := getRunningPods(ctx, r.Cli, obj)
- if err != nil {
- return nil, err
- }
- if len(podList) == 0 {
- return nil, nil
- }
- primaryPods := make([]string, 0)
- emptyRolePods := make([]string, 0)
- vertexes := make([]graph.Vertex, 0)
- for _, pod := range podList {
- role, ok := pod.Labels[constant.RoleLabelKey]
- if !ok || role == "" {
- emptyRolePods = append(emptyRolePods, pod.Name)
- continue
- }
- if role == constant.Primary {
- primaryPods = append(primaryPods, pod.Name)
- }
- }
-
- for i := range podList {
- pod := &podList[i]
- needUpdate := false
- if pod.Annotations == nil {
- pod.Annotations = map[string]string{}
- }
- switch {
- case len(emptyRolePods) == len(podList):
- // if the workload is newly created, and the role label is not set, we set the pod with index=0 as the primary by default.
- needUpdate = handlePrimaryNotExistPod(pod)
- default:
- if len(primaryPods) != 1 {
- return nil, errors.New(fmt.Sprintf("the number of primary pod is not equal to 1, primary pods: %v, emptyRole pods: %v", primaryPods, emptyRolePods))
- }
- needUpdate = handlePrimaryExistPod(pod, primaryPods[0])
- }
- if needUpdate {
- vertexes = append(vertexes, &ictrltypes.LifecycleVertex{
- Obj: pod,
- Action: ictrltypes.ActionPatchPtr(),
- })
- }
- }
- // rebuild cluster.status.components.replicationSet.status
- if err := rebuildReplicationSetClusterStatus(r.Cluster, r.getWorkloadType(), r.getName(), podList); err != nil {
- return nil, err
- }
- return vertexes, nil
-}
-
-// handlePrimaryNotExistPod is used to handle the pod which is not exists primary pod.
-func handlePrimaryNotExistPod(pod *corev1.Pod) bool {
- parent, o := ParseParentNameAndOrdinal(pod.Name)
- defaultRole := DefaultRole(o)
- pod.GetLabels()[constant.RoleLabelKey] = defaultRole
- pod.Annotations[constant.PrimaryAnnotationKey] = fmt.Sprintf("%s-%d", parent, 0)
- return true
-}
-
-// handlePrimaryExistPod is used to handle the pod which is exists primary pod.
-func handlePrimaryExistPod(pod *corev1.Pod, primary string) bool {
- needPatch := false
- if pod.Name != primary {
- role, ok := pod.Labels[constant.RoleLabelKey]
- if !ok || role == "" {
- pod.GetLabels()[constant.RoleLabelKey] = constant.Secondary
- needPatch = true
- }
- }
- pk, ok := pod.Annotations[constant.PrimaryAnnotationKey]
- if !ok || pk != primary {
- pod.Annotations[constant.PrimaryAnnotationKey] = primary
- needPatch = true
- }
- return needPatch
-}
-
-// DefaultRole is used to get the default role of the Pod of the Replication workload.
-func DefaultRole(i int32) string {
- role := constant.Secondary
- if i == 0 {
- role = constant.Primary
- }
- return role
-}
-
-// newReplicationSet is the constructor of the type replicationSet.
-func newReplicationSet(cli client.Client,
- cluster *appsv1alpha1.Cluster,
- spec *appsv1alpha1.ClusterComponentSpec,
- def appsv1alpha1.ClusterComponentDefinition) *replicationSet {
- return &replicationSet{
- stateful: stateful{
- componentSetBase: componentSetBase{
- Cli: cli,
- Cluster: cluster,
- SynthesizedComponent: nil,
- ComponentSpec: spec,
- ComponentDef: &def,
- },
- },
- }
-}
diff --git a/controllers/apps/components/replication_set_test.go b/controllers/apps/components/replication_set_test.go
deleted file mode 100644
index 095db7ceb56..00000000000
--- a/controllers/apps/components/replication_set_test.go
+++ /dev/null
@@ -1,282 +0,0 @@
-/*
-Copyright (C) 2022-2023 ApeCloud Co., Ltd
-
-This file is part of KubeBlocks project
-
-This program is free software: you can redistribute it and/or modify
-it under the terms of the GNU Affero General Public License as published by
-the Free Software Foundation, either version 3 of the License, or
-(at your option) any later version.
-
-This program is distributed in the hope that it will be useful
-but WITHOUT ANY WARRANTY; without even the implied warranty of
-MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
-GNU Affero General Public License for more details.
-
-You should have received a copy of the GNU Affero General Public License
-along with this program. If not, see .
-*/
-
-package components
-
-import (
- "fmt"
-
- . "github.com/onsi/ginkgo/v2"
- . "github.com/onsi/gomega"
-
- appsv1 "k8s.io/api/apps/v1"
- corev1 "k8s.io/api/core/v1"
- "sigs.k8s.io/controller-runtime/pkg/client"
-
- appsv1alpha1 "github.com/apecloud/kubeblocks/apis/apps/v1alpha1"
- "github.com/apecloud/kubeblocks/internal/constant"
- ictrltypes "github.com/apecloud/kubeblocks/internal/controller/types"
- intctrlutil "github.com/apecloud/kubeblocks/internal/generics"
- testapps "github.com/apecloud/kubeblocks/internal/testutil/apps"
- testk8s "github.com/apecloud/kubeblocks/internal/testutil/k8s"
-)
-
-var _ = Describe("Replication Component", func() {
- var (
- clusterName = "test-cluster-repl"
- clusterDefName = "test-cluster-def-repl"
- clusterVersionName = "test-cluster-version-repl"
- controllerRivision = "mock-revision"
- )
-
- var (
- clusterDefObj *appsv1alpha1.ClusterDefinition
- clusterVersionObj *appsv1alpha1.ClusterVersion
- clusterObj *appsv1alpha1.Cluster
- )
-
- cleanAll := func() {
- // must wait till resources deleted and no longer existed before the testcases start,
- // otherwise if later it needs to create some new resource objects with the same name,
- // in race conditions, it will find the existence of old objects, resulting failure to
- // create the new objects.
- By("clean resources")
- // delete cluster(and all dependent sub-resources), clusterversion and clusterdef
- testapps.ClearClusterResources(&testCtx)
-
- // clear rest resources
- inNS := client.InNamespace(testCtx.DefaultNamespace)
- ml := client.HasLabels{testCtx.TestObjLabelKey}
- // namespaced resources
- testapps.ClearResources(&testCtx, intctrlutil.StatefulSetSignature, inNS, ml)
- testapps.ClearResources(&testCtx, intctrlutil.PodSignature, inNS, ml, client.GracePeriodSeconds(0))
- }
-
- BeforeEach(cleanAll)
-
- AfterEach(cleanAll)
-
- Context("Replication Component test", func() {
- It("Replication Component test", func() {
-
- By("Create a clusterDefinition obj with replication workloadType.")
- replicationSpec := &appsv1alpha1.ReplicationSetSpec{
- StatefulSetSpec: appsv1alpha1.StatefulSetSpec{
- UpdateStrategy: appsv1alpha1.SerialStrategy,
- },
- }
- clusterDefObj = testapps.NewClusterDefFactory(clusterDefName).
- AddComponentDef(testapps.ReplicationRedisComponent, testapps.DefaultRedisCompDefName).
- AddReplicationSpec(replicationSpec).
- Create(&testCtx).GetObject()
-
- By("Create a clusterVersion obj with replication workloadType.")
- clusterVersionObj = testapps.NewClusterVersionFactory(clusterVersionName, clusterDefObj.Name).
- AddComponentVersion(testapps.DefaultRedisCompDefName).AddContainerShort(testapps.DefaultRedisContainerName, testapps.DefaultRedisImageName).
- Create(&testCtx).GetObject()
-
- By("Creating a cluster with replication workloadType.")
- clusterObj = testapps.NewClusterFactory(testCtx.DefaultNamespace, clusterName,
- clusterDefObj.Name, clusterVersionObj.Name).WithRandomName().
- AddComponent(testapps.DefaultRedisCompSpecName, testapps.DefaultRedisCompDefName).
- SetReplicas(testapps.DefaultReplicationReplicas).
- Create(&testCtx).GetObject()
-
- // mock cluster is Running
- Expect(testapps.ChangeObjStatus(&testCtx, clusterObj, func() {
- clusterObj.Status.Components = map[string]appsv1alpha1.ClusterComponentStatus{
- testapps.DefaultRedisCompSpecName: {
- Phase: appsv1alpha1.RunningClusterCompPhase,
- },
- }
- })).Should(Succeed())
-
- By("Creating statefulSet of replication workloadType.")
- replicas := int32(2)
- status := appsv1.StatefulSetStatus{
- AvailableReplicas: replicas,
- ObservedGeneration: 1,
- Replicas: replicas,
- ReadyReplicas: replicas,
- UpdatedReplicas: replicas,
- CurrentRevision: controllerRivision,
- UpdateRevision: controllerRivision,
- }
-
- replicationSetSts := testapps.NewStatefulSetFactory(testCtx.DefaultNamespace,
- clusterObj.Name+"-"+testapps.DefaultRedisCompSpecName, clusterObj.Name, testapps.DefaultRedisCompSpecName).
- AddContainer(corev1.Container{Name: testapps.DefaultRedisContainerName, Image: testapps.DefaultRedisImageName}).
- AddAppInstanceLabel(clusterObj.Name).
- AddAppComponentLabel(testapps.DefaultRedisCompSpecName).
- AddAppManangedByLabel().
- SetReplicas(replicas).
- Create(&testCtx).GetObject()
- stsObjectKey := client.ObjectKey{Name: replicationSetSts.Name, Namespace: testCtx.DefaultNamespace}
-
- Expect(replicationSetSts.Spec.VolumeClaimTemplates).Should(BeEmpty())
-
- compDefName := clusterObj.Spec.GetComponentDefRefName(testapps.DefaultRedisCompSpecName)
- componentDef := clusterDefObj.GetComponentDefByName(compDefName)
- componentSpec := clusterObj.Spec.GetComponentByName(testapps.DefaultRedisCompSpecName)
- replicationComponent := newReplicationSet(k8sClient, clusterObj, componentSpec, *componentDef)
- var podList []*corev1.Pod
-
- for _, availableReplica := range []int32{0, replicas} {
- status.AvailableReplicas = availableReplica
- replicationSetSts.Status = status
- testk8s.PatchStatefulSetStatus(&testCtx, replicationSetSts.Name, status)
-
- if availableReplica > 0 {
- // Create pods of the statefulset
- stsPods := testapps.MockReplicationComponentPods(nil, testCtx, replicationSetSts, clusterObj.Name,
- testapps.DefaultRedisCompSpecName, map[int32]string{
- 0: constant.Primary,
- 1: constant.Secondary,
- })
- podList = append(podList, stsPods...)
- By("Testing pods are ready")
- podsReady, _ := replicationComponent.PodsReady(ctx, replicationSetSts)
- Expect(podsReady).Should(BeTrue())
- By("Testing component is running")
- isRunning, _ := replicationComponent.IsRunning(ctx, replicationSetSts)
- Expect(isRunning).Should(BeTrue())
- } else {
- podsReady, _ := replicationComponent.PodsReady(ctx, replicationSetSts)
- By("Testing pods are not ready")
- Expect(podsReady).Should(BeFalse())
- By("Testing component is not running")
- isRunning, _ := replicationComponent.IsRunning(ctx, replicationSetSts)
- Expect(isRunning).Should(BeFalse())
- }
- }
-
- // TODO(refactor): probe timed-out pod
- // By("Testing handle probe timed out")
- // requeue, _ := replicationComponent.HandleProbeTimeoutWhenPodsReady(ctx, nil)
- // Expect(requeue == false).Should(BeTrue())
-
- By("Testing pod is available")
- primaryPod := podList[0]
- Expect(replicationComponent.PodIsAvailable(primaryPod, 10)).Should(BeTrue())
-
- By("should return empty string if pod of component is only not ready when component is not up running")
- pod := podList[1]
- Expect(testapps.ChangeObjStatus(&testCtx, pod, func() {
- pod.Status.Conditions = []corev1.PodCondition{}
- })).Should(Succeed())
- status.AvailableReplicas -= 1
- testk8s.PatchStatefulSetStatus(&testCtx, replicationSetSts.Name, status)
- phase, _, _ := replicationComponent.GetPhaseWhenPodsNotReady(ctx, testapps.DefaultRedisCompSpecName, false)
- Expect(string(phase)).Should(Equal(""))
-
- By("expect component phase is Abnormal when pod of component is not ready and component is up running")
- phase, _, _ = replicationComponent.GetPhaseWhenPodsNotReady(ctx, testapps.DefaultRedisCompSpecName, true)
- Expect(phase).Should(Equal(appsv1alpha1.AbnormalClusterCompPhase))
-
- // mock pod label is empty
- Expect(testapps.ChangeObj(&testCtx, primaryPod, func(pod *corev1.Pod) {
- pod.Labels[constant.RoleLabelKey] = ""
- })).Should(Succeed())
- _, statusMessages, _ := replicationComponent.GetPhaseWhenPodsNotReady(ctx, testapps.DefaultRedisCompSpecName, false)
- Expect(statusMessages[fmt.Sprintf("%s/%s", primaryPod.Kind, primaryPod.Name)]).
- Should(ContainSubstring("empty label for pod, please check"))
-
- // mock primary pod failed
- testk8s.UpdatePodStatusScheduleFailed(ctx, testCtx, primaryPod.Name, primaryPod.Namespace)
- phase, _, _ = replicationComponent.GetPhaseWhenPodsNotReady(ctx, testapps.DefaultRedisCompSpecName, true)
- Expect(phase).Should(Equal(appsv1alpha1.FailedClusterCompPhase))
-
- By("Checking if the pod is not updated when statefulSet is not updated")
- Expect(testCtx.Cli.Get(testCtx.Ctx, stsObjectKey, replicationSetSts)).Should(Succeed())
- vertexes, err := replicationComponent.HandleRestart(ctx, replicationSetSts)
- Expect(err).To(Succeed())
- Expect(len(vertexes)).To(Equal(0))
- pods, err := GetPodListByStatefulSet(ctx, k8sClient, replicationSetSts)
- Expect(err).To(Succeed())
- Expect(len(pods)).To(Equal(int(replicas)))
- Expect(isStsAndPodsRevisionConsistent(ctx, k8sClient, replicationSetSts)).Should(BeTrue())
-
- By("Checking if the pod is deleted when statefulSet is updated and UpdateStrategy is SerialStrategy")
- status.UpdateRevision = "new-mock-revision"
- testk8s.PatchStatefulSetStatus(&testCtx, replicationSetSts.Name, status)
- Expect(testCtx.Cli.Get(testCtx.Ctx, stsObjectKey, replicationSetSts)).Should(Succeed())
- vertexes, err = replicationComponent.HandleRestart(ctx, replicationSetSts)
- Expect(err).To(Succeed())
- Expect(len(vertexes)).To(Equal(1))
- Expect(*vertexes[0].(*ictrltypes.LifecycleVertex).Action == ictrltypes.DELETE).To(BeTrue())
-
- By("Checking if the pod is deleted when statefulSet is updated and UpdateStrategy is BestEffortParallelStrategy")
- Expect(testapps.ChangeObj(&testCtx, clusterDefObj, func(clusterDef *appsv1alpha1.ClusterDefinition) {
- clusterDef.Spec.ComponentDefs[0].ReplicationSpec = &appsv1alpha1.ReplicationSetSpec{
- StatefulSetSpec: appsv1alpha1.StatefulSetSpec{
- UpdateStrategy: appsv1alpha1.BestEffortParallelStrategy,
- },
- }
- })).Should(Succeed())
- status.UpdateRevision = "new-mock-revision-2"
- testk8s.PatchStatefulSetStatus(&testCtx, replicationSetSts.Name, status)
- Expect(testCtx.Cli.Get(testCtx.Ctx, stsObjectKey, replicationSetSts)).Should(Succeed())
- vertexes, err = replicationComponent.HandleRestart(ctx, replicationSetSts)
- Expect(err).To(Succeed())
- Expect(len(vertexes)).To(Equal(1))
- Expect(*vertexes[0].(*ictrltypes.LifecycleVertex).Action == ictrltypes.DELETE).To(BeTrue())
-
- By("Checking if the pod is deleted when statefulSet is updated and UpdateStrategy is ParallelStrategy")
- Expect(testapps.ChangeObj(&testCtx, clusterDefObj, func(clusterDef *appsv1alpha1.ClusterDefinition) {
- clusterDef.Spec.ComponentDefs[0].ReplicationSpec = &appsv1alpha1.ReplicationSetSpec{
- StatefulSetSpec: appsv1alpha1.StatefulSetSpec{
- UpdateStrategy: appsv1alpha1.ParallelStrategy,
- },
- }
- })).Should(Succeed())
- status.UpdateRevision = "new-mock-revision-2"
- testk8s.PatchStatefulSetStatus(&testCtx, replicationSetSts.Name, status)
- Expect(testCtx.Cli.Get(testCtx.Ctx, stsObjectKey, replicationSetSts)).Should(Succeed())
- vertexes, err = replicationComponent.HandleRestart(ctx, replicationSetSts)
- Expect(err).To(Succeed())
- Expect(len(vertexes)).To(Equal(2))
- Expect(*vertexes[0].(*ictrltypes.LifecycleVertex).Action == ictrltypes.DELETE).To(BeTrue())
-
- By("Test handleRoleChange when statefulSet Pod with role label but without primary annotation")
- Expect(testapps.ChangeObj(&testCtx, primaryPod, func(pod *corev1.Pod) {
- pod.Labels[constant.RoleLabelKey] = constant.Primary
- })).Should(Succeed())
- status.UpdateRevision = "new-mock-revision-for-role-change"
- testk8s.PatchStatefulSetStatus(&testCtx, replicationSetSts.Name, status)
- Expect(testCtx.Cli.Get(testCtx.Ctx, stsObjectKey, replicationSetSts)).Should(Succeed())
- vertexes, err = replicationComponent.HandleRoleChange(ctx, replicationSetSts)
- Expect(err).To(Succeed())
- Expect(len(vertexes)).To(Equal(int(replicas)))
- Expect(*vertexes[0].(*ictrltypes.LifecycleVertex).Action == ictrltypes.PATCH).To(BeTrue())
-
- By("Test handleRoleChange when statefulSet h-scale out a new Pod with no role label")
- status.Replicas = 3
- status.AvailableReplicas = 3
- status.ReadyReplicas = 3
- testk8s.PatchStatefulSetStatus(&testCtx, replicationSetSts.Name, status)
- Expect(testCtx.Cli.Get(testCtx.Ctx, stsObjectKey, replicationSetSts)).Should(Succeed())
- newPodName := fmt.Sprintf("%s-%d", replicationSetSts.Name, 2)
- _ = testapps.MockReplicationComponentPod(nil, testCtx, replicationSetSts, clusterObj.Name, testapps.DefaultRedisCompSpecName, newPodName, "")
- vertexes, err = replicationComponent.HandleRoleChange(ctx, replicationSetSts)
- Expect(err).To(Succeed())
- Expect(len(vertexes)).To(Equal(3))
- Expect(*vertexes[0].(*ictrltypes.LifecycleVertex).Action == ictrltypes.PATCH).To(BeTrue())
- })
- })
-})
diff --git a/controllers/apps/components/replication_set_utils.go b/controllers/apps/components/replication_set_utils.go
deleted file mode 100644
index f985a677aad..00000000000
--- a/controllers/apps/components/replication_set_utils.go
+++ /dev/null
@@ -1,214 +0,0 @@
-/*
-Copyright (C) 2022-2023 ApeCloud Co., Ltd
-
-This file is part of KubeBlocks project
-
-This program is free software: you can redistribute it and/or modify
-it under the terms of the GNU Affero General Public License as published by
-the Free Software Foundation, either version 3 of the License, or
-(at your option) any later version.
-
-This program is distributed in the hope that it will be useful
-but WITHOUT ANY WARRANTY; without even the implied warranty of
-MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
-GNU Affero General Public License for more details.
-
-You should have received a copy of the GNU Affero General Public License
-along with this program. If not, see .
-*/
-
-package components
-
-import (
- "context"
- "fmt"
- "time"
-
- "github.com/google/go-cmp/cmp"
- "golang.org/x/exp/slices"
- corev1 "k8s.io/api/core/v1"
- "sigs.k8s.io/controller-runtime/pkg/client"
-
- appsv1alpha1 "github.com/apecloud/kubeblocks/apis/apps/v1alpha1"
- "github.com/apecloud/kubeblocks/internal/constant"
- intctrlutil "github.com/apecloud/kubeblocks/internal/controllerutil"
- "github.com/apecloud/kubeblocks/internal/generics"
-)
-
-// rebuildReplicationSetClusterStatus syncs replicationSet pod status to cluster.status.component[componentName].ReplicationStatus.
-func rebuildReplicationSetClusterStatus(cluster *appsv1alpha1.Cluster,
- workloadType appsv1alpha1.WorkloadType, compName string, podList []corev1.Pod) error {
- if len(podList) == 0 {
- return nil
- }
-
- var oldReplicationStatus *appsv1alpha1.ReplicationSetStatus
- if v, ok := cluster.Status.Components[compName]; ok {
- oldReplicationStatus = v.ReplicationSetStatus
- }
-
- newReplicationStatus := &appsv1alpha1.ReplicationSetStatus{}
- if err := genReplicationSetStatus(newReplicationStatus, podList); err != nil {
- return err
- }
- // if status changed, do update
- if !cmp.Equal(newReplicationStatus, oldReplicationStatus) {
- if err := initClusterComponentStatusIfNeed(cluster, compName, workloadType); err != nil {
- return err
- }
- componentStatus := cluster.Status.Components[compName]
- componentStatus.ReplicationSetStatus = newReplicationStatus
- cluster.Status.SetComponentStatus(compName, componentStatus)
- }
- return nil
-}
-
-// genReplicationSetStatus generates ReplicationSetStatus from podList.
-func genReplicationSetStatus(replicationStatus *appsv1alpha1.ReplicationSetStatus, podList []corev1.Pod) error {
- for _, pod := range podList {
- role := pod.Labels[constant.RoleLabelKey]
- if role == "" {
- return fmt.Errorf("pod %s has no role label", pod.Name)
- }
- switch role {
- case constant.Primary:
- if replicationStatus.Primary.Pod != "" {
- return fmt.Errorf("more than one primary pod found")
- }
- replicationStatus.Primary.Pod = pod.Name
- case constant.Secondary:
- replicationStatus.Secondaries = append(replicationStatus.Secondaries, appsv1alpha1.ReplicationMemberStatus{
- Pod: pod.Name,
- })
- default:
- return fmt.Errorf("unknown role %s", role)
- }
- }
- return nil
-}
-
-// updateObjRoleChangedInfo updates the value of the role label and annotation of the object.
-func updateObjRoleChangedInfo[T generics.Object, PT generics.PObject[T]](
- ctx context.Context, cli client.Client, event *corev1.Event, obj T, role string) error {
- pObj := PT(&obj)
- patch := client.MergeFrom(PT(pObj.DeepCopy()))
- pObj.GetLabels()[constant.RoleLabelKey] = role
- if pObj.GetAnnotations() == nil {
- pObj.SetAnnotations(map[string]string{})
- }
- pObj.GetAnnotations()[constant.LastRoleChangedEventTimestampAnnotationKey] = event.EventTime.Time.Format(time.RFC3339Nano)
- if err := cli.Patch(ctx, pObj, patch); err != nil {
- return err
- }
- return nil
-}
-
-// HandleReplicationSetRoleChangeEvent handles the role change event of the replication workload when switchPolicy is Noop.
-func HandleReplicationSetRoleChangeEvent(cli client.Client,
- reqCtx intctrlutil.RequestCtx,
- event *corev1.Event,
- cluster *appsv1alpha1.Cluster,
- compName string,
- pod *corev1.Pod,
- newRole string) error {
- reqCtx.Log.Info("receive role change event", "podName", pod.Name, "current pod role label", pod.Labels[constant.RoleLabelKey], "new role", newRole)
- // if newRole is not Primary or Secondary, ignore it.
- if !slices.Contains([]string{constant.Primary, constant.Secondary}, newRole) {
- reqCtx.Log.Info("replicationSet new role is invalid, please check", "new role", newRole)
- return nil
- }
-
- // if switchPolicy is not Noop, return
- clusterCompSpec := getClusterComponentSpecByName(*cluster, compName)
- if clusterCompSpec == nil || clusterCompSpec.SwitchPolicy == nil || clusterCompSpec.SwitchPolicy.Type != appsv1alpha1.Noop {
- reqCtx.Log.Info("cluster switchPolicy is not Noop, does not support handling role change event", "cluster", cluster.Name)
- return nil
- }
-
- // update pod role label with newRole
- if err := updateObjRoleChangedInfo(reqCtx.Ctx, cli, event, *pod, newRole); err != nil {
- reqCtx.Log.Info("failed to update pod role label", "podName", pod.Name, "newRole", newRole, "err", err)
- return err
- }
- reqCtx.Log.Info("succeed to update pod role label", "podName", pod.Name, "newRole", newRole)
- return nil
-}
-
-// composeReplicationRolePriorityMap generates a priority map based on roles.
-func composeReplicationRolePriorityMap() map[string]int {
- return map[string]int{
- "": emptyReplicationPriority,
- constant.Primary: primaryPriority,
- constant.Secondary: secondaryPriority,
- }
-}
-
-// generateReplicationParallelPlan generates a parallel plan for the replication workload.
-// unknown & empty & secondary & primary
-func generateReplicationParallelPlan(plan *Plan, pods []corev1.Pod, rolePriorityMap map[string]int) {
- start := plan.Start
- for _, pod := range pods {
- nextStep := &Step{}
- nextStep.Obj = pod
- start.NextSteps = append(start.NextSteps, nextStep)
- }
-}
-
-// generateReplicationSerialPlan generates a serial plan for the replication workload.
-// unknown -> empty -> secondary -> primary
-func generateReplicationSerialPlan(plan *Plan, pods []corev1.Pod, rolePriorityMap map[string]int) {
- start := plan.Start
- for _, pod := range pods {
- nextStep := &Step{}
- nextStep.Obj = pod
- start.NextSteps = append(start.NextSteps, nextStep)
- start = nextStep
- }
-}
-
-// generateReplicationBestEffortParallelPlan generates a best effort parallel plan for the replication workload.
-// unknown & empty & 1/2 secondaries -> 1/2 secondaries -> primary
-func generateReplicationBestEffortParallelPlan(plan *Plan, pods []corev1.Pod, rolePriorityMap map[string]int) {
- start := plan.Start
- l := len(pods)
- unknownEmptySteps := make([]*Step, 0, l)
- secondarySteps := make([]*Step, 0, l)
- primarySteps := make([]*Step, 0, l)
-
- for _, pod := range pods {
- role := pod.Labels[constant.RoleLabelKey]
- nextStep := &Step{Obj: pod}
- switch {
- case rolePriorityMap[role] <= emptyReplicationPriority:
- unknownEmptySteps = append(unknownEmptySteps, nextStep)
- case rolePriorityMap[role] < primaryPriority:
- secondarySteps = append(secondarySteps, nextStep)
- default:
- primarySteps = append(primarySteps, nextStep)
- }
- }
-
- // append unknown, empty
- if len(unknownEmptySteps) > 0 {
- start.NextSteps = append(start.NextSteps, unknownEmptySteps...)
- start = start.NextSteps[0]
- }
-
- // append 1/2 secondaries
- end := len(secondarySteps) / 2
- if end > 0 {
- start.NextSteps = append(start.NextSteps, secondarySteps[:end]...)
- start = start.NextSteps[0]
- }
-
- // append the other 1/2 secondaries
- if len(secondarySteps) > end {
- start.NextSteps = append(start.NextSteps, secondarySteps[end:]...)
- start = start.NextSteps[0]
- }
-
- // append primary
- if len(primarySteps) > 0 {
- start.NextSteps = append(start.NextSteps, primarySteps...)
- }
-}
diff --git a/controllers/apps/components/replication_set_utils_test.go b/controllers/apps/components/replication_set_utils_test.go
deleted file mode 100644
index d50e8edb53b..00000000000
--- a/controllers/apps/components/replication_set_utils_test.go
+++ /dev/null
@@ -1,278 +0,0 @@
-/*
-Copyright (C) 2022-2023 ApeCloud Co., Ltd
-
-This file is part of KubeBlocks project
-
-This program is free software: you can redistribute it and/or modify
-it under the terms of the GNU Affero General Public License as published by
-the Free Software Foundation, either version 3 of the License, or
-(at your option) any later version.
-
-This program is distributed in the hope that it will be useful
-but WITHOUT ANY WARRANTY; without even the implied warranty of
-MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
-GNU Affero General Public License for more details.
-
-You should have received a copy of the GNU Affero General Public License
-along with this program. If not, see .
-*/
-
-package components
-
-import (
- "context"
- "fmt"
- "strings"
- "time"
-
- . "github.com/onsi/ginkgo/v2"
- . "github.com/onsi/gomega"
-
- "github.com/sethvargo/go-password/password"
- corev1 "k8s.io/api/core/v1"
- metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
- "k8s.io/apimachinery/pkg/types"
- "sigs.k8s.io/controller-runtime/pkg/client"
- "sigs.k8s.io/controller-runtime/pkg/log"
-
- appsv1alpha1 "github.com/apecloud/kubeblocks/apis/apps/v1alpha1"
- "github.com/apecloud/kubeblocks/internal/constant"
- "github.com/apecloud/kubeblocks/internal/controller/builder"
- intctrlutil "github.com/apecloud/kubeblocks/internal/controllerutil"
- "github.com/apecloud/kubeblocks/internal/generics"
- testapps "github.com/apecloud/kubeblocks/internal/testutil/apps"
- testk8s "github.com/apecloud/kubeblocks/internal/testutil/k8s"
- lorryutil "github.com/apecloud/kubeblocks/lorry/util"
-)
-
-var _ = Describe("replicationSet Util", func() {
-
- var (
- clusterName = "test-cluster-repl"
- clusterDefName = "test-cluster-def-repl"
- clusterVersionName = "test-cluster-version-repl"
- )
-
- var (
- clusterDefObj *appsv1alpha1.ClusterDefinition
- clusterVersionObj *appsv1alpha1.ClusterVersion
- clusterObj *appsv1alpha1.Cluster
- )
-
- cleanAll := func() {
- // must wait till resources deleted and no longer existed before the testcases start,
- // otherwise if later it needs to create some new resource objects with the same name,
- // in race conditions, it will find the existence of old objects, resulting failure to
- // create the new objects.
- By("clean resources")
- // delete cluster(and all dependent sub-resources), clusterversion and clusterdef
- testapps.ClearClusterResources(&testCtx)
-
- // clear rest resources
- inNS := client.InNamespace(testCtx.DefaultNamespace)
- ml := client.HasLabels{testCtx.TestObjLabelKey}
- // namespaced resources
- testapps.ClearResourcesWithRemoveFinalizerOption(&testCtx, generics.StatefulSetSignature, true, inNS, ml)
- testapps.ClearResources(&testCtx, generics.PodSignature, inNS, ml, client.GracePeriodSeconds(0))
- }
-
- BeforeEach(cleanAll)
-
- AfterEach(cleanAll)
-
- testHandleReplicationSet := func() {
-
- By("Creating a cluster with replication workloadType.")
- clusterObj = testapps.NewClusterFactory(testCtx.DefaultNamespace, clusterName,
- clusterDefObj.Name, clusterVersionObj.Name).WithRandomName().
- AddComponent(testapps.DefaultRedisCompSpecName, testapps.DefaultRedisCompDefName).
- SetReplicas(testapps.DefaultReplicationReplicas).
- Create(&testCtx).GetObject()
-
- By("Creating a statefulSet of replication workloadType.")
- container := corev1.Container{
- Name: "mock-redis-container",
- Image: testapps.DefaultRedisImageName,
- ImagePullPolicy: corev1.PullIfNotPresent,
- }
- sts := testapps.NewStatefulSetFactory(testCtx.DefaultNamespace,
- clusterObj.Name+"-"+testapps.DefaultRedisCompSpecName, clusterObj.Name, testapps.DefaultRedisCompSpecName).
- AddFinalizers([]string{constant.DBClusterFinalizerName}).
- AddContainer(container).
- AddAppInstanceLabel(clusterObj.Name).
- AddAppComponentLabel(testapps.DefaultRedisCompSpecName).
- AddAppManangedByLabel().
- SetReplicas(2).
- Create(&testCtx).GetObject()
-
- By("Creating Pods of replication workloadType.")
- for i := int32(0); i < *sts.Spec.Replicas; i++ {
- _ = testapps.NewPodFactory(testCtx.DefaultNamespace, fmt.Sprintf("%s-%d", sts.Name, i)).
- AddContainer(container).
- AddLabelsInMap(sts.Labels).
- AddRoleLabel(DefaultRole(i)).
- Create(&testCtx).GetObject()
- }
- }
-
- testNeedUpdateReplicationSetStatus := func() {
- By("Creating a cluster with replication workloadType.")
- clusterObj = testapps.NewClusterFactory(testCtx.DefaultNamespace, clusterName,
- clusterDefObj.Name, clusterVersionObj.Name).WithRandomName().
- AddComponent(testapps.DefaultRedisCompSpecName, testapps.DefaultRedisCompDefName).Create(&testCtx).GetObject()
-
- By("init replicationSet cluster status")
- patch := client.MergeFrom(clusterObj.DeepCopy())
- clusterObj.Status.Phase = appsv1alpha1.RunningClusterPhase
- clusterObj.Status.Components = map[string]appsv1alpha1.ClusterComponentStatus{
- testapps.DefaultRedisCompSpecName: {
- Phase: appsv1alpha1.RunningClusterCompPhase,
- ReplicationSetStatus: &appsv1alpha1.ReplicationSetStatus{
- Primary: appsv1alpha1.ReplicationMemberStatus{
- Pod: clusterObj.Name + testapps.DefaultRedisCompSpecName + "-0",
- },
- Secondaries: []appsv1alpha1.ReplicationMemberStatus{
- {
- Pod: clusterObj.Name + testapps.DefaultRedisCompSpecName + "-1",
- },
- {
- Pod: clusterObj.Name + testapps.DefaultRedisCompSpecName + "-2",
- },
- },
- },
- },
- }
- Expect(k8sClient.Status().Patch(context.Background(), clusterObj, patch)).Should(Succeed())
-
- By("testing sync cluster status with add pod")
-
- var podList []corev1.Pod
- sts := testk8s.NewFakeStatefulSet(clusterObj.Name+testapps.DefaultRedisCompSpecName, 4)
-
- for i := int32(0); i < *sts.Spec.Replicas; i++ {
- pod := testapps.NewPodFactory(testCtx.DefaultNamespace, fmt.Sprintf("%s-%d", sts.Name, i)).
- AddContainer(corev1.Container{Name: testapps.DefaultRedisContainerName, Image: testapps.DefaultRedisImageName}).
- AddRoleLabel(DefaultRole(i)).
- Create(&testCtx).GetObject()
- podList = append(podList, *pod)
- }
- err := genReplicationSetStatus(clusterObj.Status.Components[testapps.DefaultRedisCompSpecName].ReplicationSetStatus, podList)
- Expect(err).ShouldNot(Succeed())
- Expect(err.Error()).Should(ContainSubstring("more than one primary pod found"))
-
- newReplicationStatus := &appsv1alpha1.ReplicationSetStatus{}
- Expect(genReplicationSetStatus(newReplicationStatus, podList)).Should(Succeed())
- Expect(len(newReplicationStatus.Secondaries)).Should(Equal(3))
- }
-
- createRoleChangedEvent := func(podName, role string, podUid types.UID) *corev1.Event {
- seq, _ := password.Generate(16, 16, 0, true, true)
- objectRef := corev1.ObjectReference{
- APIVersion: "v1",
- Kind: "Pod",
- Namespace: testCtx.DefaultNamespace,
- Name: podName,
- UID: podUid,
- }
- eventName := strings.Join([]string{podName, seq}, ".")
- return builder.NewEventBuilder(testCtx.DefaultNamespace, eventName).
- SetInvolvedObject(objectRef).
- SetMessage(fmt.Sprintf("{\"event\":\"roleChanged\",\"originalRole\":\"secondary\",\"role\":\"%s\"}", role)).
- SetReason(string(lorryutil.CheckRoleOperation)).
- SetType(corev1.EventTypeNormal).
- SetFirstTimestamp(metav1.NewTime(time.Now())).
- SetLastTimestamp(metav1.NewTime(time.Now())).
- GetObject()
- }
-
- testHandleReplicationSetRoleChangeEvent := func() {
- By("Creating a cluster with replication workloadType.")
- clusterSwitchPolicy := &appsv1alpha1.ClusterSwitchPolicy{
- Type: appsv1alpha1.Noop,
- }
- clusterObj = testapps.NewClusterFactory(testCtx.DefaultNamespace, clusterName,
- clusterDefObj.Name, clusterVersionObj.Name).WithRandomName().
- AddComponent(testapps.DefaultRedisCompSpecName, testapps.DefaultRedisCompDefName).
- SetReplicas(testapps.DefaultReplicationReplicas).
- SetSwitchPolicy(clusterSwitchPolicy).
- Create(&testCtx).GetObject()
-
- By("Creating a statefulSet of replication workloadType.")
- container := corev1.Container{
- Name: "mock-redis-container",
- Image: testapps.DefaultRedisImageName,
- ImagePullPolicy: corev1.PullIfNotPresent,
- }
- sts := testapps.NewStatefulSetFactory(testCtx.DefaultNamespace,
- clusterObj.Name+"-"+testapps.DefaultRedisCompSpecName, clusterObj.Name, testapps.DefaultRedisCompSpecName).
- AddContainer(container).
- AddAppInstanceLabel(clusterObj.Name).
- AddAppComponentLabel(testapps.DefaultRedisCompSpecName).
- AddAppManangedByLabel().
- SetReplicas(2).
- Create(&testCtx).GetObject()
-
- By("Creating Pods of replication workloadType.")
- var (
- primaryPod *corev1.Pod
- secondaryPods []*corev1.Pod
- )
- for i := int32(0); i < *sts.Spec.Replicas; i++ {
- pod := testapps.NewPodFactory(testCtx.DefaultNamespace, fmt.Sprintf("%s-%d", sts.Name, i)).
- AddContainer(container).
- AddLabelsInMap(sts.Labels).
- AddRoleLabel(DefaultRole(i)).
- Create(&testCtx).GetObject()
- if pod.Labels[constant.RoleLabelKey] == constant.Primary {
- primaryPod = pod
- } else {
- secondaryPods = append(secondaryPods, pod)
- }
- }
- Expect(primaryPod).ShouldNot(BeNil())
- Expect(secondaryPods).ShouldNot(BeEmpty())
-
- By("Test update replicationSet pod role label with event driver, secondary change to primary.")
- reqCtx := intctrlutil.RequestCtx{
- Ctx: testCtx.Ctx,
- Log: log.FromContext(ctx).WithValues("event", testCtx.DefaultNamespace),
- }
- event := createRoleChangedEvent(secondaryPods[0].Name, constant.Primary, secondaryPods[0].UID)
- Expect(HandleReplicationSetRoleChangeEvent(k8sClient, reqCtx, event, clusterObj, testapps.DefaultRedisCompSpecName,
- secondaryPods[0], constant.Primary)).ShouldNot(HaveOccurred())
-
- By("Test when secondary change to primary, the old primary label has been updated at the same time, so return nil directly.")
- event = createRoleChangedEvent(primaryPod.Name, constant.Secondary, primaryPod.UID)
- Expect(HandleReplicationSetRoleChangeEvent(k8sClient, reqCtx, event, clusterObj, testapps.DefaultRedisCompSpecName,
- primaryPod, constant.Secondary)).ShouldNot(HaveOccurred())
- }
-
- // Scenarios
-
- Context("test replicationSet util", func() {
- BeforeEach(func() {
- By("Create a clusterDefinition obj with replication workloadType.")
- clusterDefObj = testapps.NewClusterDefFactory(clusterDefName).
- AddComponentDef(testapps.ReplicationRedisComponent, testapps.DefaultRedisCompDefName).
- Create(&testCtx).GetObject()
-
- By("Create a clusterVersion obj with replication workloadType.")
- clusterVersionObj = testapps.NewClusterVersionFactory(clusterVersionName, clusterDefObj.GetName()).
- AddComponentVersion(testapps.DefaultRedisCompDefName).AddContainerShort(testapps.DefaultRedisContainerName, testapps.DefaultRedisImageName).
- Create(&testCtx).GetObject()
-
- })
-
- It("Test handReplicationSet with different conditions", func() {
- testHandleReplicationSet()
- })
-
- It("Test need update replicationSet status when horizontal scaling adds pod or removes pod", func() {
- testNeedUpdateReplicationSetStatus()
- })
-
- It("Test update pod role label by roleChangedEvent when ha switch", func() {
- testHandleReplicationSetRoleChangeEvent()
- })
- })
-})
diff --git a/controllers/apps/components/replication_workload.go b/controllers/apps/components/replication_workload.go
deleted file mode 100644
index d65f78d6b20..00000000000
--- a/controllers/apps/components/replication_workload.go
+++ /dev/null
@@ -1,53 +0,0 @@
-/*
-Copyright (C) 2022-2023 ApeCloud Co., Ltd
-
-This file is part of KubeBlocks project
-
-This program is free software: you can redistribute it and/or modify
-it under the terms of the GNU Affero General Public License as published by
-the Free Software Foundation, either version 3 of the License, or
-(at your option) any later version.
-
-This program is distributed in the hope that it will be useful
-but WITHOUT ANY WARRANTY; without even the implied warranty of
-MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
-GNU Affero General Public License for more details.
-
-You should have received a copy of the GNU Affero General Public License
-along with this program. If not, see .
-*/
-
-package components
-
-import (
- "github.com/apecloud/kubeblocks/internal/controller/factory"
- "sigs.k8s.io/controller-runtime/pkg/client"
-
- "github.com/apecloud/kubeblocks/internal/constant"
-)
-
-type replicationComponentWorkloadBuilder struct {
- componentWorkloadBuilderBase
-}
-
-var _ componentWorkloadBuilder = &replicationComponentWorkloadBuilder{}
-
-func (b *replicationComponentWorkloadBuilder) BuildWorkload() componentWorkloadBuilder {
- return b.BuildWorkload4StatefulSet("replication")
-}
-
-func (b *replicationComponentWorkloadBuilder) BuildService() componentWorkloadBuilder {
- buildFn := func() ([]client.Object, error) {
- svcList, err := factory.BuildSvcList(b.Comp.GetCluster(), b.Comp.GetSynthesizedComponent())
- if err != nil {
- return nil, err
- }
- objs := make([]client.Object, 0, len(svcList))
- for _, svc := range svcList {
- svc.Spec.Selector[constant.RoleLabelKey] = constant.Primary
- objs = append(objs, svc)
- }
- return objs, err
- }
- return b.BuildWrapper(buildFn)
-}
diff --git a/controllers/apps/components/rsm.go b/controllers/apps/components/rsm.go
deleted file mode 100644
index fe8eb6ddf57..00000000000
--- a/controllers/apps/components/rsm.go
+++ /dev/null
@@ -1,107 +0,0 @@
-/*
-Copyright (C) 2022-2023 ApeCloud Co., Ltd
-
-This file is part of KubeBlocks project
-
-This program is free software: you can redistribute it and/or modify
-it under the terms of the GNU Affero General Public License as published by
-the Free Software Foundation, either version 3 of the License, or
-(at your option) any later version.
-
-This program is distributed in the hope that it will be useful
-but WITHOUT ANY WARRANTY; without even the implied warranty of
-MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
-GNU Affero General Public License for more details.
-
-You should have received a copy of the GNU Affero General Public License
-along with this program. If not, see .
-*/
-
-package components
-
-import (
- "k8s.io/client-go/tools/record"
- "sigs.k8s.io/controller-runtime/pkg/client"
-
- appsv1alpha1 "github.com/apecloud/kubeblocks/apis/apps/v1alpha1"
- "github.com/apecloud/kubeblocks/internal/controller/component"
- "github.com/apecloud/kubeblocks/internal/controller/graph"
- ictrltypes "github.com/apecloud/kubeblocks/internal/controller/types"
- intctrlutil "github.com/apecloud/kubeblocks/internal/controllerutil"
-)
-
-type rsmComponent struct {
- rsmComponentBase
-}
-
-var _ Component = &rsmComponent{}
-
-const workloadType = "RSM"
-
-func newRSMComponent(cli client.Client,
- recorder record.EventRecorder,
- cluster *appsv1alpha1.Cluster,
- clusterVersion *appsv1alpha1.ClusterVersion,
- synthesizedComponent *component.SynthesizedComponent,
- dag *graph.DAG) *rsmComponent {
- comp := &rsmComponent{
- rsmComponentBase: rsmComponentBase{
- componentBase: componentBase{
- Client: cli,
- Recorder: recorder,
- Cluster: cluster,
- ClusterVersion: clusterVersion,
- Component: synthesizedComponent,
- ComponentSet: &RSM{
- componentSetBase: componentSetBase{
- Cli: cli,
- Cluster: cluster,
- SynthesizedComponent: synthesizedComponent,
- ComponentSpec: nil,
- ComponentDef: nil,
- },
- },
- Dag: dag,
- WorkloadVertex: nil,
- },
- },
- }
- return comp
-}
-
-func (c *rsmComponent) newBuilder(reqCtx intctrlutil.RequestCtx, cli client.Client,
- action *ictrltypes.LifecycleAction) componentWorkloadBuilder {
- builder := &rsmComponentWorkloadBuilder{
- componentWorkloadBuilderBase: componentWorkloadBuilderBase{
- ReqCtx: reqCtx,
- Client: cli,
- Comp: c,
- DefaultAction: action,
- Error: nil,
- EnvConfig: nil,
- Workload: nil,
- },
- }
- builder.ConcreteBuilder = builder
- return builder
-}
-
-func (c *rsmComponent) GetWorkloadType() appsv1alpha1.WorkloadType {
- return workloadType
-}
-
-func (c *rsmComponent) GetBuiltObjects(reqCtx intctrlutil.RequestCtx, cli client.Client) ([]client.Object, error) {
- return c.rsmComponentBase.GetBuiltObjects(c.newBuilder(reqCtx, cli, ictrltypes.ActionCreatePtr()))
-}
-
-func (c *rsmComponent) Create(reqCtx intctrlutil.RequestCtx, cli client.Client) error {
- return c.rsmComponentBase.Create(reqCtx, cli, c.newBuilder(reqCtx, cli, ictrltypes.ActionCreatePtr()))
-}
-
-func (c *rsmComponent) Update(reqCtx intctrlutil.RequestCtx, cli client.Client) error {
- return c.rsmComponentBase.Update(reqCtx, cli, c.newBuilder(reqCtx, cli, nil))
-}
-
-func (c *rsmComponent) Status(reqCtx intctrlutil.RequestCtx, cli client.Client) error {
- return c.rsmComponentBase.Status(reqCtx, cli, c.newBuilder(reqCtx, cli, ictrltypes.ActionNoopPtr()))
-}
diff --git a/controllers/apps/components/rsm_set.go b/controllers/apps/components/rsm_set.go
deleted file mode 100644
index f7a86efba24..00000000000
--- a/controllers/apps/components/rsm_set.go
+++ /dev/null
@@ -1,337 +0,0 @@
-/*
-Copyright (C) 2022-2023 ApeCloud Co., Ltd
-
-This file is part of KubeBlocks project
-
-This program is free software: you can redistribute it and/or modify
-it under the terms of the GNU Affero General Public License as published by
-the Free Software Foundation, either version 3 of the License, or
-(at your option) any later version.
-
-This program is distributed in the hope that it will be useful
-but WITHOUT ANY WARRANTY; without even the implied warranty of
-MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
-GNU Affero General Public License for more details.
-
-You should have received a copy of the GNU Affero General Public License
-along with this program. If not, see .
-*/
-
-package components
-
-import (
- "context"
- "fmt"
- "time"
-
- "github.com/google/go-cmp/cmp"
- corev1 "k8s.io/api/core/v1"
- metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
- "k8s.io/kubectl/pkg/util/podutils"
- "sigs.k8s.io/controller-runtime/pkg/client"
- "sigs.k8s.io/controller-runtime/pkg/log"
-
- appsv1alpha1 "github.com/apecloud/kubeblocks/apis/apps/v1alpha1"
- workloads "github.com/apecloud/kubeblocks/apis/workloads/v1alpha1"
- "github.com/apecloud/kubeblocks/internal/constant"
- "github.com/apecloud/kubeblocks/internal/controller/component"
- "github.com/apecloud/kubeblocks/internal/controller/graph"
- rsmcore "github.com/apecloud/kubeblocks/internal/controller/rsm"
- ictrltypes "github.com/apecloud/kubeblocks/internal/controller/types"
- intctrlutil "github.com/apecloud/kubeblocks/internal/controllerutil"
-)
-
-type RSM struct {
- componentSetBase
-}
-
-var _ componentSet = &RSM{}
-
-func (r *RSM) getName() string {
- if r.SynthesizedComponent != nil {
- return r.SynthesizedComponent.Name
- }
- return r.ComponentSpec.Name
-}
-
-func (r *RSM) getReplicas() int32 {
- if r.SynthesizedComponent != nil {
- return r.SynthesizedComponent.Replicas
- }
- return r.ComponentSpec.Replicas
-}
-
-func (r *RSM) IsRunning(ctx context.Context, obj client.Object) (bool, error) {
- if obj == nil {
- return false, nil
- }
- rsm, ok := obj.(*workloads.ReplicatedStateMachine)
- if !ok {
- return false, nil
- }
- if isLatestRevision, err := IsComponentPodsWithLatestRevision(ctx, r.Cli, r.Cluster, rsm); err != nil {
- return false, err
- } else if !isLatestRevision {
- return false, nil
- }
-
- // whether rsm is ready
- return rsmcore.IsRSMReady(rsm), nil
-}
-
-func (r *RSM) PodsReady(ctx context.Context, obj client.Object) (bool, error) {
- if obj == nil {
- return false, nil
- }
- rsm, ok := obj.(*workloads.ReplicatedStateMachine)
- if !ok {
- return false, nil
- }
- sts := ConvertRSMToSTS(rsm)
- return statefulSetPodsAreReady(sts, r.getReplicas()), nil
-}
-
-func (r *RSM) PodIsAvailable(pod *corev1.Pod, minReadySeconds int32) bool {
- switch {
- case pod == nil:
- return false
- case !podutils.IsPodAvailable(pod, minReadySeconds, metav1.Time{Time: time.Now()}):
- return false
- case r.SynthesizedComponent.WorkloadType == appsv1alpha1.Consensus,
- r.SynthesizedComponent.WorkloadType == appsv1alpha1.Replication:
- return intctrlutil.PodIsReadyWithLabel(*pod)
- default:
- return true
- }
-}
-
-func (r *RSM) GetPhaseWhenPodsReadyAndProbeTimeout(pods []*corev1.Pod) (appsv1alpha1.ClusterComponentPhase, appsv1alpha1.ComponentMessageMap) {
- if r.SynthesizedComponent.WorkloadType != appsv1alpha1.Consensus {
- return "", nil
- }
-
- var (
- isAbnormal bool
- isFailed = true
- statusMessages appsv1alpha1.ComponentMessageMap
- )
- getProbes := func() *appsv1alpha1.ClusterDefinitionProbes {
- if r.SynthesizedComponent != nil {
- return r.SynthesizedComponent.Probes
- }
- return r.ComponentDef.Probes
- }
- getConsensusSpec := func() *appsv1alpha1.ConsensusSetSpec {
- if r.SynthesizedComponent != nil {
- return r.SynthesizedComponent.ConsensusSpec
- }
- return r.ComponentDef.ConsensusSpec
- }
- compStatus, ok := r.Cluster.Status.Components[r.getName()]
- if !ok || compStatus.PodsReadyTime == nil {
- return "", nil
- }
- if !isProbeTimeout(getProbes(), compStatus.PodsReadyTime) {
- return "", nil
- }
- for _, pod := range pods {
- role := pod.Labels[constant.RoleLabelKey]
- if role == getConsensusSpec().Leader.Name {
- isFailed = false
- }
- if role == "" {
- isAbnormal = true
- statusMessages.SetObjectMessage(pod.Kind, pod.Name, "Role probe timeout, check whether the application is available")
- }
- // TODO clear up the message of ready pod in component.message.
- }
- switch {
- case isFailed:
- return appsv1alpha1.FailedClusterCompPhase, statusMessages
- case isAbnormal:
- return appsv1alpha1.AbnormalClusterCompPhase, statusMessages
- default:
- return "", statusMessages
- }
-}
-
-// GetPhaseWhenPodsNotReady gets the component phase when the pods of component are not ready.
-func (r *RSM) GetPhaseWhenPodsNotReady(ctx context.Context,
- componentName string,
- originPhaseIsUpRunning bool) (appsv1alpha1.ClusterComponentPhase, appsv1alpha1.ComponentMessageMap, error) {
- rsmList := &workloads.ReplicatedStateMachineList{}
- podList, err := getCompRelatedObjectList(ctx, r.Cli, *r.Cluster, componentName, rsmList)
- if err != nil || len(rsmList.Items) == 0 {
- return "", nil, err
- }
- statusMessages := appsv1alpha1.ComponentMessageMap{}
- // if the failed pod is not controlled by the latest revision
- podIsControlledByLatestRevision := func(pod *corev1.Pod, rsm *workloads.ReplicatedStateMachine) bool {
- return rsm.Status.ObservedGeneration == rsm.Generation && intctrlutil.GetPodRevision(pod) == rsm.Status.UpdateRevision
- }
- checkLeaderIsReady := func(pod *corev1.Pod, workload metav1.Object) bool {
- getLeaderRoleName := func() string {
- switch r.SynthesizedComponent.WorkloadType {
- case appsv1alpha1.Consensus:
- return r.SynthesizedComponent.ConsensusSpec.Leader.Name
- case appsv1alpha1.Replication:
- return constant.Primary
- default:
- return ""
- }
- }
- leaderRoleName := getLeaderRoleName()
- labelValue := pod.Labels[constant.RoleLabelKey]
- return labelValue == leaderRoleName && intctrlutil.PodIsReady(pod)
- }
- checkExistFailedPodOfLatestRevision := func(pod *corev1.Pod, workload metav1.Object) bool {
- rsm := workload.(*workloads.ReplicatedStateMachine)
- // if component is up running but pod is not ready, this pod should be failed.
- // for example: full disk cause readiness probe failed and serve is not available.
- // but kubelet only sets the container is not ready and pod is also Running.
- if originPhaseIsUpRunning {
- return !intctrlutil.PodIsReady(pod) && podIsControlledByLatestRevision(pod, rsm)
- }
- isFailed, _, message := IsPodFailedAndTimedOut(pod)
- existLatestRevisionFailedPod := isFailed && podIsControlledByLatestRevision(pod, rsm)
- if existLatestRevisionFailedPod {
- statusMessages.SetObjectMessage(pod.Kind, pod.Name, message)
- }
- return existLatestRevisionFailedPod
- }
- rsmObj := rsmList.Items[0]
- return getComponentPhaseWhenPodsNotReady(podList, &rsmObj, r.getReplicas(),
- rsmObj.Status.AvailableReplicas, checkLeaderIsReady, checkExistFailedPodOfLatestRevision), statusMessages, nil
-}
-
-func (r *RSM) HandleRestart(context.Context, client.Object) ([]graph.Vertex, error) {
- return nil, nil
-}
-
-func (r *RSM) HandleRoleChange(ctx context.Context, obj client.Object) ([]graph.Vertex, error) {
- if r.SynthesizedComponent.WorkloadType != appsv1alpha1.Consensus &&
- r.SynthesizedComponent.WorkloadType != appsv1alpha1.Replication {
- return nil, nil
- }
-
- // update cluster.status.component.consensusSetStatus based on the existences for all pods
- componentName := r.getName()
- rsmObj, _ := obj.(*workloads.ReplicatedStateMachine)
- switch r.SynthesizedComponent.WorkloadType {
- case appsv1alpha1.Consensus:
- // first, get the old status
- var oldConsensusSetStatus *appsv1alpha1.ConsensusSetStatus
- if v, ok := r.Cluster.Status.Components[componentName]; ok {
- oldConsensusSetStatus = v.ConsensusSetStatus
- }
- // create the initial status
- newConsensusSetStatus := &appsv1alpha1.ConsensusSetStatus{
- Leader: appsv1alpha1.ConsensusMemberStatus{
- Name: "",
- Pod: constant.ComponentStatusDefaultPodName,
- AccessMode: appsv1alpha1.None,
- },
- }
- // then, set the new status
- setConsensusSetStatusRolesByRSM(newConsensusSetStatus, rsmObj)
- // if status changed, do update
- if !cmp.Equal(newConsensusSetStatus, oldConsensusSetStatus) {
- if err := initClusterComponentStatusIfNeed(r.Cluster, componentName, appsv1alpha1.Consensus); err != nil {
- return nil, err
- }
- componentStatus := r.Cluster.Status.Components[componentName]
- componentStatus.ConsensusSetStatus = newConsensusSetStatus
- r.Cluster.Status.SetComponentStatus(componentName, componentStatus)
-
- return nil, nil
- }
- case appsv1alpha1.Replication:
- sts := ConvertRSMToSTS(rsmObj)
- podList, err := getRunningPods(ctx, r.Cli, sts)
- if err != nil {
- return nil, err
- }
- if len(podList) == 0 {
- return nil, nil
- }
- primaryPods := make([]string, 0)
- emptyRolePods := make([]string, 0)
- vertexes := make([]graph.Vertex, 0)
- for _, pod := range podList {
- role, ok := pod.Labels[constant.RoleLabelKey]
- if !ok || role == "" {
- emptyRolePods = append(emptyRolePods, pod.Name)
- continue
- }
- if role == constant.Primary {
- primaryPods = append(primaryPods, pod.Name)
- }
- }
-
- for i := range podList {
- pod := &podList[i]
- needUpdate := false
- if pod.Annotations == nil {
- pod.Annotations = map[string]string{}
- }
- switch {
- case len(emptyRolePods) == len(podList):
- // if the workload is newly created, and the role label is not set, we set the pod with index=0 as the primary by default.
- needUpdate = handlePrimaryNotExistPod(pod)
- default:
- if len(primaryPods) != 1 {
- return nil, fmt.Errorf("the number of primary pod is not equal to 1, primary pods: %v, emptyRole pods: %v", primaryPods, emptyRolePods)
- }
- needUpdate = handlePrimaryExistPod(pod, primaryPods[0])
- }
- if needUpdate {
- vertexes = append(vertexes, &ictrltypes.LifecycleVertex{
- Obj: pod,
- Action: ictrltypes.ActionPatchPtr(),
- })
- }
- }
- // rebuild cluster.status.components.replicationSet.status
- if err := rebuildReplicationSetClusterStatus(r.Cluster, appsv1alpha1.Replication, componentName, podList); err != nil {
- return nil, err
- }
- return vertexes, nil
- }
-
- return nil, nil
-}
-
-func setConsensusSetStatusRolesByRSM(newConsensusSetStatus *appsv1alpha1.ConsensusSetStatus, rsmObj *workloads.ReplicatedStateMachine) {
- for _, memberStatus := range rsmObj.Status.MembersStatus {
- status := appsv1alpha1.ConsensusMemberStatus{
- Name: memberStatus.Name,
- Pod: memberStatus.PodName,
- AccessMode: appsv1alpha1.AccessMode(memberStatus.AccessMode),
- }
- switch {
- case memberStatus.IsLeader:
- newConsensusSetStatus.Leader = status
- case memberStatus.CanVote:
- newConsensusSetStatus.Followers = append(newConsensusSetStatus.Followers, status)
- default:
- newConsensusSetStatus.Learner = &status
- }
- }
-}
-
-func newRSM(ctx context.Context,
- cli client.Client,
- cluster *appsv1alpha1.Cluster,
- clusterDef *appsv1alpha1.ClusterDefinition,
- spec *appsv1alpha1.ClusterComponentSpec,
- def appsv1alpha1.ClusterComponentDefinition) *RSM {
- reqCtx := intctrlutil.RequestCtx{Log: log.FromContext(ctx).WithValues("rsm-test", def.Name)}
- synthesizedComponent, _ := component.BuildComponent(reqCtx, nil, cluster, clusterDef, &def, spec, nil)
- return &RSM{
- componentSetBase: componentSetBase{
- Cli: cli,
- Cluster: cluster,
- SynthesizedComponent: synthesizedComponent,
- },
- }
-}
diff --git a/controllers/apps/components/rsm_set_test.go b/controllers/apps/components/rsm_set_test.go
deleted file mode 100644
index 8f1d262caed..00000000000
--- a/controllers/apps/components/rsm_set_test.go
+++ /dev/null
@@ -1,212 +0,0 @@
-/*
-Copyright (C) 2022-2023 ApeCloud Co., Ltd
-
-This file is part of KubeBlocks project
-
-This program is free software: you can redistribute it and/or modify
-it under the terms of the GNU Affero General Public License as published by
-the Free Software Foundation, either version 3 of the License, or
-(at your option) any later version.
-
-This program is distributed in the hope that it will be useful
-but WITHOUT ANY WARRANTY; without even the implied warranty of
-MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
-GNU Affero General Public License for more details.
-
-You should have received a copy of the GNU Affero General Public License
-along with this program. If not, see .
-*/
-
-package components
-
-import (
- "fmt"
- "strconv"
- "time"
-
- . "github.com/onsi/ginkgo/v2"
- . "github.com/onsi/gomega"
-
- appsv1 "k8s.io/api/apps/v1"
- corev1 "k8s.io/api/core/v1"
- metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
- "sigs.k8s.io/controller-runtime/pkg/client"
-
- appsv1alpha1 "github.com/apecloud/kubeblocks/apis/apps/v1alpha1"
- workloads "github.com/apecloud/kubeblocks/apis/workloads/v1alpha1"
- "github.com/apecloud/kubeblocks/internal/constant"
- intctrlutil "github.com/apecloud/kubeblocks/internal/generics"
- testapps "github.com/apecloud/kubeblocks/internal/testutil/apps"
- testk8s "github.com/apecloud/kubeblocks/internal/testutil/k8s"
-)
-
-var _ = Describe("RSM Component", func() {
- var (
- randomStr = testCtx.GetRandomStr()
- clusterDefName = "mysql1-clusterdef-" + randomStr
- clusterVersionName = "mysql1-clusterversion-" + randomStr
- clusterName = "mysql1-" + randomStr
- )
- const (
- defaultMinReadySeconds = 10
- rsmCompDefRef = "stateful"
- rsmCompName = "stateful"
- )
- cleanAll := func() {
- // must wait till resources deleted and no longer existed before the testcases start,
- // otherwise if later it needs to create some new resource objects with the same name,
- // in race conditions, it will find the existence of old objects, resulting failure to
- // create the new objects.
- By("clean resources")
- // delete cluster(and all dependent sub-resources), clusterversion and clusterdef
- testapps.ClearClusterResources(&testCtx)
-
- // clear rest resources
- inNS := client.InNamespace(testCtx.DefaultNamespace)
- ml := client.HasLabels{testCtx.TestObjLabelKey}
- // namespaced resources
- testapps.ClearResources(&testCtx, intctrlutil.StatefulSetSignature, inNS, ml)
- testapps.ClearResources(&testCtx, intctrlutil.PodSignature, inNS, ml, client.GracePeriodSeconds(0))
- }
-
- BeforeEach(cleanAll)
-
- AfterEach(cleanAll)
-
- Context("RSM Component test", func() {
- It("RSM Component test", func() {
- By(" init cluster, statefulSet, pods")
- clusterDef, _, cluster := testapps.InitConsensusMysql(&testCtx, clusterDefName,
- clusterVersionName, clusterName, rsmCompDefRef, rsmCompName)
- rsm := testapps.MockRSMComponent(&testCtx, clusterName, rsmCompName)
- Expect(testapps.ChangeObj(&testCtx, rsm, func(machine *workloads.ReplicatedStateMachine) {
- annotations := machine.Annotations
- if annotations == nil {
- annotations = make(map[string]string, 0)
- }
- annotations[constant.KubeBlocksGenerationKey] = strconv.FormatInt(cluster.Generation, 10)
- machine.Annotations = annotations
- })).Should(Succeed())
- Expect(testapps.ChangeObjStatus(&testCtx, cluster, func() {
- cluster.Status.ObservedGeneration = cluster.Generation
- })).Should(Succeed())
- rsmList := &workloads.ReplicatedStateMachineList{}
- Eventually(func() bool {
- _ = k8sClient.List(ctx, rsmList, client.InNamespace(testCtx.DefaultNamespace), client.MatchingLabels{
- constant.AppInstanceLabelKey: clusterName,
- constant.KBAppComponentLabelKey: rsmCompName,
- }, client.Limit(1))
- return len(rsmList.Items) > 0
- }).Should(BeTrue())
- _ = testapps.MockConsensusComponentStatefulSet(&testCtx, clusterName, rsmCompName)
- stsList := &appsv1.StatefulSetList{}
- Eventually(func() bool {
- _ = k8sClient.List(ctx, stsList, client.InNamespace(testCtx.DefaultNamespace), client.MatchingLabels{
- constant.AppInstanceLabelKey: clusterName,
- constant.KBAppComponentLabelKey: rsmCompName,
- }, client.Limit(1))
- return len(stsList.Items) > 0
- }).Should(BeTrue())
-
- By("test pods number of sts is 0")
- rsm = &rsmList.Items[0]
- clusterComponent := cluster.Spec.GetComponentByName(rsmCompName)
- componentDef := clusterDef.GetComponentDefByName(clusterComponent.ComponentDefRef)
- rsmComponent := newRSM(testCtx.Ctx, k8sClient, cluster, clusterDef, clusterComponent, *componentDef)
- phase, _, _ := rsmComponent.GetPhaseWhenPodsNotReady(ctx, rsmCompName, false)
- Expect(phase == appsv1alpha1.FailedClusterCompPhase).Should(BeTrue())
-
- By("test pods are not ready")
- updateRevision := fmt.Sprintf("%s-%s-%s", clusterName, rsmCompName, "6fdd48d9cd")
- sts := &stsList.Items[0]
- Expect(testapps.ChangeObjStatus(&testCtx, sts, func() {
- availableReplicas := *sts.Spec.Replicas - 1
- sts.Status.AvailableReplicas = availableReplicas
- sts.Status.ReadyReplicas = availableReplicas
- sts.Status.Replicas = availableReplicas
- sts.Status.ObservedGeneration = 1
- sts.Status.UpdateRevision = updateRevision
- })).Should(Succeed())
- Expect(testapps.ChangeObjStatus(&testCtx, rsm, func() {
- availableReplicas := *rsm.Spec.Replicas - 1
- rsm.Status.InitReplicas = *rsm.Spec.Replicas
- rsm.Status.AvailableReplicas = availableReplicas
- rsm.Status.ReadyReplicas = availableReplicas
- rsm.Status.Replicas = availableReplicas
- rsm.Status.ObservedGeneration = 1
- rsm.Status.CurrentGeneration = 1
- rsm.Status.UpdateRevision = updateRevision
- })).Should(Succeed())
- podsReady, _ := rsmComponent.PodsReady(ctx, rsm)
- Expect(podsReady).Should(BeFalse())
-
- By("create pods of sts")
- podList := testapps.MockConsensusComponentPods(&testCtx, sts, clusterName, rsmCompName)
-
- By("test rsm component is abnormal")
- pod := podList[0]
- // mock pod is not ready
- Expect(testapps.ChangeObjStatus(&testCtx, pod, func() {
- pod.Status.Conditions = []corev1.PodCondition{}
- })).Should(Succeed())
- Eventually(testapps.CheckObj(&testCtx, client.ObjectKeyFromObject(rsm), func(g Gomega, tmpRSM *workloads.ReplicatedStateMachine) {
- g.Expect(tmpRSM.Status.AvailableReplicas == *rsm.Spec.Replicas-1).Should(BeTrue())
- })).Should(Succeed())
-
- By("should return empty string if pod of component is only not ready when component is not up running")
- phase, _, _ = rsmComponent.GetPhaseWhenPodsNotReady(ctx, rsmCompName, false)
- Expect(string(phase)).Should(Equal(""))
-
- By("expect component phase is Failed when pod of component is not ready and component is up running")
- phase, _, _ = rsmComponent.GetPhaseWhenPodsNotReady(ctx, rsmCompName, true)
- Expect(phase).Should(Equal(appsv1alpha1.FailedClusterCompPhase))
-
- By("expect component phase is Failed when pod of component is failed")
- testk8s.UpdatePodStatusScheduleFailed(ctx, testCtx, pod.Name, pod.Namespace)
- phase, _, _ = rsmComponent.GetPhaseWhenPodsNotReady(ctx, rsmCompName, false)
- Expect(phase).Should(Equal(appsv1alpha1.FailedClusterCompPhase))
-
- By("not ready pod is not controlled by latest revision, should return empty string")
- // mock pod is not controlled by latest revision
- Expect(testapps.ChangeObj(&testCtx, pod, func(lpod *corev1.Pod) {
- lpod.Labels[appsv1.ControllerRevisionHashLabelKey] = fmt.Sprintf("%s-%s-%s", clusterName, rsmCompName, "5wdsd8d9fs")
- })).Should(Succeed())
- phase, _, _ = rsmComponent.GetPhaseWhenPodsNotReady(ctx, rsmCompName, false)
- Expect(string(phase)).Should(Equal(""))
- // reset updateRevision
- Expect(testapps.ChangeObj(&testCtx, pod, func(lpod *corev1.Pod) {
- lpod.Labels[appsv1.ControllerRevisionHashLabelKey] = updateRevision
- })).Should(Succeed())
-
- By("test pod is available")
- lastTransTime := metav1.NewTime(time.Now().Add(-1 * (defaultMinReadySeconds + 1) * time.Second))
- testk8s.MockPodAvailable(pod, lastTransTime)
- Expect(rsmComponent.PodIsAvailable(pod, defaultMinReadySeconds)).Should(BeTrue())
-
- By("test pods are ready")
- // mock sts is ready
- testk8s.MockStatefulSetReady(sts)
- testk8s.MockRSMReady(rsm, podList...)
- Eventually(func() bool {
- podsReady, _ = rsmComponent.PodsReady(ctx, rsm)
- return podsReady
- }).Should(BeTrue())
-
- By("test component.replicas is inconsistent with rsm.spec.replicas")
- oldReplicas := rsmComponent.SynthesizedComponent.Replicas
- replicas := int32(4)
- rsmComponent.SynthesizedComponent.Replicas = replicas
- rsm.Annotations[constant.KubeBlocksGenerationKey] = "new-generation"
- isRunning, _ := rsmComponent.IsRunning(ctx, rsm)
- Expect(isRunning).Should(BeFalse())
- // reset replicas
- rsmComponent.SynthesizedComponent.Replicas = oldReplicas
- rsm.Annotations[constant.KubeBlocksGenerationKey] = strconv.FormatInt(cluster.Generation, 10)
-
- By("test component is running")
- isRunning, _ = rsmComponent.IsRunning(ctx, rsm)
- Expect(isRunning).Should(BeTrue())
- })
- })
-
-})
diff --git a/controllers/apps/components/stateful.go b/controllers/apps/components/stateful.go
deleted file mode 100644
index cb4d94cf5bb..00000000000
--- a/controllers/apps/components/stateful.go
+++ /dev/null
@@ -1,105 +0,0 @@
-/*
-Copyright (C) 2022-2023 ApeCloud Co., Ltd
-
-This file is part of KubeBlocks project
-
-This program is free software: you can redistribute it and/or modify
-it under the terms of the GNU Affero General Public License as published by
-the Free Software Foundation, either version 3 of the License, or
-(at your option) any later version.
-
-This program is distributed in the hope that it will be useful
-but WITHOUT ANY WARRANTY; without even the implied warranty of
-MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
-GNU Affero General Public License for more details.
-
-You should have received a copy of the GNU Affero General Public License
-along with this program. If not, see .
-*/
-
-package components
-
-import (
- "k8s.io/client-go/tools/record"
- "sigs.k8s.io/controller-runtime/pkg/client"
-
- appsv1alpha1 "github.com/apecloud/kubeblocks/apis/apps/v1alpha1"
- "github.com/apecloud/kubeblocks/internal/controller/component"
- "github.com/apecloud/kubeblocks/internal/controller/graph"
- ictrltypes "github.com/apecloud/kubeblocks/internal/controller/types"
- intctrlutil "github.com/apecloud/kubeblocks/internal/controllerutil"
-)
-
-func newStatefulComponent(cli client.Client,
- recorder record.EventRecorder,
- cluster *appsv1alpha1.Cluster,
- clusterVersion *appsv1alpha1.ClusterVersion,
- synthesizedComponent *component.SynthesizedComponent,
- dag *graph.DAG) *statefulComponent {
- comp := &statefulComponent{
- statefulComponentBase: statefulComponentBase{
- componentBase: componentBase{
- Client: cli,
- Recorder: recorder,
- Cluster: cluster,
- ClusterVersion: clusterVersion,
- Component: synthesizedComponent,
- ComponentSet: &stateful{
- componentSetBase: componentSetBase{
- Cli: cli,
- Cluster: cluster,
- SynthesizedComponent: synthesizedComponent,
- ComponentSpec: nil,
- ComponentDef: nil,
- },
- },
- Dag: dag,
- WorkloadVertex: nil,
- },
- },
- }
- return comp
-}
-
-type statefulComponent struct {
- statefulComponentBase
-}
-
-var _ Component = &statefulComponent{}
-
-func (c *statefulComponent) newBuilder(reqCtx intctrlutil.RequestCtx, cli client.Client,
- action *ictrltypes.LifecycleAction) componentWorkloadBuilder {
- builder := &statefulComponentWorkloadBuilder{
- componentWorkloadBuilderBase: componentWorkloadBuilderBase{
- ReqCtx: reqCtx,
- Client: cli,
- Comp: c,
- DefaultAction: action,
- Error: nil,
- EnvConfig: nil,
- Workload: nil,
- },
- }
- builder.ConcreteBuilder = builder
- return builder
-}
-
-func (c *statefulComponent) GetWorkloadType() appsv1alpha1.WorkloadType {
- return appsv1alpha1.Stateful
-}
-
-func (c *statefulComponent) GetBuiltObjects(reqCtx intctrlutil.RequestCtx, cli client.Client) ([]client.Object, error) {
- return c.statefulComponentBase.GetBuiltObjects(c.newBuilder(reqCtx, cli, ictrltypes.ActionCreatePtr()))
-}
-
-func (c *statefulComponent) Create(reqCtx intctrlutil.RequestCtx, cli client.Client) error {
- return c.statefulComponentBase.Create(reqCtx, cli, c.newBuilder(reqCtx, cli, ictrltypes.ActionCreatePtr()))
-}
-
-func (c *statefulComponent) Update(reqCtx intctrlutil.RequestCtx, cli client.Client) error {
- return c.statefulComponentBase.Update(reqCtx, cli, c.newBuilder(reqCtx, cli, nil))
-}
-
-func (c *statefulComponent) Status(reqCtx intctrlutil.RequestCtx, cli client.Client) error {
- return c.statefulComponentBase.Status(reqCtx, cli, c.newBuilder(reqCtx, cli, ictrltypes.ActionNoopPtr()))
-}
diff --git a/controllers/apps/components/stateful_set.go b/controllers/apps/components/stateful_set.go
deleted file mode 100644
index 38420fe6304..00000000000
--- a/controllers/apps/components/stateful_set.go
+++ /dev/null
@@ -1,274 +0,0 @@
-/*
-Copyright (C) 2022-2023 ApeCloud Co., Ltd
-
-This file is part of KubeBlocks project
-
-This program is free software: you can redistribute it and/or modify
-it under the terms of the GNU Affero General Public License as published by
-the Free Software Foundation, either version 3 of the License, or
-(at your option) any later version.
-
-This program is distributed in the hope that it will be useful
-but WITHOUT ANY WARRANTY; without even the implied warranty of
-MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
-GNU Affero General Public License for more details.
-
-You should have received a copy of the GNU Affero General Public License
-along with this program. If not, see .
-*/
-
-package components
-
-import (
- "context"
- "errors"
- "time"
-
- appsv1 "k8s.io/api/apps/v1"
- corev1 "k8s.io/api/core/v1"
- metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
- "k8s.io/kubectl/pkg/util/podutils"
- "sigs.k8s.io/controller-runtime/pkg/client"
-
- appsv1alpha1 "github.com/apecloud/kubeblocks/apis/apps/v1alpha1"
- "github.com/apecloud/kubeblocks/internal/constant"
- "github.com/apecloud/kubeblocks/internal/controller/graph"
- ictrltypes "github.com/apecloud/kubeblocks/internal/controller/types"
- intctrlutil "github.com/apecloud/kubeblocks/internal/controllerutil"
-)
-
-type stateful struct {
- componentSetBase
-}
-
-var _ componentSet = &stateful{}
-
-func (r *stateful) getReplicas() int32 {
- if r.SynthesizedComponent != nil {
- return r.SynthesizedComponent.Replicas
- }
- return r.ComponentSpec.Replicas
-}
-
-func (r *stateful) IsRunning(ctx context.Context, obj client.Object) (bool, error) {
- if obj == nil {
- return false, nil
- }
- sts := convertToStatefulSet(obj)
- isRevisionConsistent, err := isStsAndPodsRevisionConsistent(ctx, r.Cli, sts)
- if err != nil {
- return false, err
- }
- targetReplicas := r.getReplicas()
- return statefulSetOfComponentIsReady(sts, isRevisionConsistent, &targetReplicas), nil
-}
-
-func (r *stateful) PodsReady(ctx context.Context, obj client.Object) (bool, error) {
- if obj == nil {
- return false, nil
- }
- sts := convertToStatefulSet(obj)
- return statefulSetPodsAreReady(sts, r.getReplicas()), nil
-}
-
-func (r *stateful) PodIsAvailable(pod *corev1.Pod, minReadySeconds int32) bool {
- if pod == nil {
- return false
- }
- return podutils.IsPodAvailable(pod, minReadySeconds, metav1.Time{Time: time.Now()})
-}
-
-func (r *stateful) GetPhaseWhenPodsReadyAndProbeTimeout(pods []*corev1.Pod) (appsv1alpha1.ClusterComponentPhase, appsv1alpha1.ComponentMessageMap) {
- return "", nil
-}
-
-// GetPhaseWhenPodsNotReady gets the component phase when the pods of component are not ready.
-func (r *stateful) GetPhaseWhenPodsNotReady(ctx context.Context,
- componentName string,
- originPhaseIsUpRunning bool) (appsv1alpha1.ClusterComponentPhase, appsv1alpha1.ComponentMessageMap, error) {
- stsList := &appsv1.StatefulSetList{}
- podList, err := getCompRelatedObjectList(ctx, r.Cli, *r.Cluster, componentName, stsList)
- if err != nil || len(stsList.Items) == 0 {
- return "", nil, err
- }
- statusMessages := appsv1alpha1.ComponentMessageMap{}
- // if the failed pod is not controlled by the latest revision
- checkExistFailedPodOfLatestRevision := func(pod *corev1.Pod, workload metav1.Object) bool {
- sts := workload.(*appsv1.StatefulSet)
- // if component is up running but pod is not ready, this pod should be failed.
- // for example: full disk cause readiness probe failed and serve is not available.
- // but kubelet only sets the container is not ready and pod is also Running.
- if originPhaseIsUpRunning {
- return !intctrlutil.PodIsReady(pod) && intctrlutil.PodIsControlledByLatestRevision(pod, sts)
- }
- isFailed, _, message := IsPodFailedAndTimedOut(pod)
- existLatestRevisionFailedPod := isFailed && intctrlutil.PodIsControlledByLatestRevision(pod, sts)
- if existLatestRevisionFailedPod {
- statusMessages.SetObjectMessage(pod.Kind, pod.Name, message)
- }
- return existLatestRevisionFailedPod
- }
- stsObj := stsList.Items[0]
- return getComponentPhaseWhenPodsNotReady(podList, &stsObj, r.getReplicas(),
- stsObj.Status.AvailableReplicas, nil, checkExistFailedPodOfLatestRevision), statusMessages, nil
-}
-
-func (r *stateful) HandleRestart(context.Context, client.Object) ([]graph.Vertex, error) {
- return nil, nil
-}
-
-func (r *stateful) HandleRoleChange(context.Context, client.Object) ([]graph.Vertex, error) {
- return nil, nil
-}
-
-// HandleUpdateWithStrategy handles the update of component with strategy.
-// REVIEW/TODO: (nashtsai)
-// 1. too many args
-func (r *stateful) HandleUpdateWithStrategy(ctx context.Context, obj client.Object,
- compStatusProcessor func(compDef *appsv1alpha1.ClusterComponentDefinition, pods []corev1.Pod, componentName string) error,
- priorityMapper func(component *appsv1alpha1.ClusterComponentDefinition) map[string]int,
- serialStrategyHandler, bestEffortParallelStrategyHandler, parallelStrategyHandler func(plan *Plan, pods []corev1.Pod, rolePriorityMap map[string]int)) ([]graph.Vertex, error) {
- if r == nil {
- return nil, nil
- }
-
- stsObj := convertToStatefulSet(obj)
- // get compDefName from stsObj.name
- compDefName := r.Cluster.Spec.GetComponentDefRefName(stsObj.Labels[constant.KBAppComponentLabelKey])
-
- // get componentDef from ClusterDefinition by compDefName
- componentDef, err := appsv1alpha1.GetComponentDefByCluster(ctx, r.Cli, *r.Cluster, compDefName)
- if err != nil {
- return nil, err
- }
-
- if componentDef == nil || componentDef.IsStatelessWorkload() {
- return nil, nil
- }
- pods, err := GetPodListByStatefulSet(ctx, r.Cli, stsObj)
- if err != nil {
- return nil, err
- }
-
- // update cluster.status.component.consensusSetStatus when all pods currently exist
- if compStatusProcessor != nil {
- componentName := stsObj.Labels[constant.KBAppComponentLabelKey]
- if err = compStatusProcessor(componentDef, pods, componentName); err != nil {
- return nil, err
- }
- }
-
- // prepare to do pods Deletion, that's the only thing we should do,
- // the statefulset reconciler will do the rest.
- // to simplify the process, we do pods Deletion after statefulset reconciliation done,
- // that is stsObj.Generation == stsObj.Status.ObservedGeneration
- if stsObj.Generation != stsObj.Status.ObservedGeneration {
- return nil, nil
- }
-
- // then we wait for all pods' presence, that is len(pods) == stsObj.Spec.Replicas
- // at that point, we have enough info about the previous pods before delete the current one
- if len(pods) != int(*stsObj.Spec.Replicas) {
- return nil, nil
- }
-
- // we don't check whether pod role label is present: prefer stateful set's Update done than role probing ready
-
- // generate the pods Deletion plan
- podsToDelete := make([]*corev1.Pod, 0)
- plan := generateUpdatePlan(stsObj, pods, componentDef, priorityMapper,
- serialStrategyHandler, bestEffortParallelStrategyHandler, parallelStrategyHandler, &podsToDelete)
- // execute plan
- if _, err := plan.WalkOneStep(); err != nil {
- return nil, err
- }
-
- vertexes := make([]graph.Vertex, 0)
- for _, pod := range podsToDelete {
- vertexes = append(vertexes, &ictrltypes.LifecycleVertex{
- Obj: pod,
- Action: ictrltypes.ActionDeletePtr(),
- Orphan: true,
- })
- }
- return vertexes, nil
-}
-
-func newStateful(cli client.Client,
- cluster *appsv1alpha1.Cluster,
- spec *appsv1alpha1.ClusterComponentSpec,
- def appsv1alpha1.ClusterComponentDefinition) *stateful {
- return &stateful{
- componentSetBase: componentSetBase{
- Cli: cli,
- Cluster: cluster,
- SynthesizedComponent: nil,
- ComponentSpec: spec,
- ComponentDef: &def,
- },
- }
-}
-
-// generateConsensusUpdatePlan generates Update plan based on UpdateStrategy
-func generateUpdatePlan(stsObj *appsv1.StatefulSet, pods []corev1.Pod,
- componentDef *appsv1alpha1.ClusterComponentDefinition,
- priorityMapper func(component *appsv1alpha1.ClusterComponentDefinition) map[string]int,
- serialStrategyHandler, bestEffortParallelStrategyHandler, parallelStrategyHandler func(plan *Plan, pods []corev1.Pod, rolePriorityMap map[string]int),
- podsToDelete *[]*corev1.Pod) *Plan {
- stsWorkload := componentDef.GetStatefulSetWorkload()
- _, s := stsWorkload.FinalStsUpdateStrategy()
- switch s.Type {
- case appsv1.RollingUpdateStatefulSetStrategyType, "":
- return nil
- }
-
- plan := &Plan{}
- plan.Start = &Step{}
- plan.WalkFunc = func(obj interface{}) (bool, error) {
- pod, ok := obj.(corev1.Pod)
- if !ok {
- return false, errors.New("wrong type: obj not Pod")
- }
-
- // if DeletionTimestamp is not nil, it is terminating.
- if pod.DeletionTimestamp != nil {
- return true, nil
- }
-
- // if pod is the latest version, we do nothing
- if intctrlutil.GetPodRevision(&pod) == stsObj.Status.UpdateRevision {
- // wait until ready
- return !intctrlutil.PodIsReadyWithLabel(pod), nil
- }
-
- // delete the pod to trigger associate StatefulSet to re-create it
- *podsToDelete = append(*podsToDelete, &pod)
-
- return true, nil
- }
-
- var rolePriorityMap map[string]int
- if priorityMapper != nil {
- rolePriorityMap = priorityMapper(componentDef)
- SortPods(pods, rolePriorityMap, constant.RoleLabelKey)
- }
-
- // generate plan by UpdateStrategy
- switch stsWorkload.GetUpdateStrategy() {
- case appsv1alpha1.ParallelStrategy:
- if parallelStrategyHandler != nil {
- parallelStrategyHandler(plan, pods, rolePriorityMap)
- }
- case appsv1alpha1.BestEffortParallelStrategy:
- if bestEffortParallelStrategyHandler != nil {
- bestEffortParallelStrategyHandler(plan, pods, rolePriorityMap)
- }
- case appsv1alpha1.SerialStrategy:
- fallthrough
- default:
- if serialStrategyHandler != nil {
- serialStrategyHandler(plan, pods, rolePriorityMap)
- }
- }
- return plan
-}
diff --git a/controllers/apps/components/stateful_set_test.go b/controllers/apps/components/stateful_set_test.go
deleted file mode 100644
index 09cba54c5a3..00000000000
--- a/controllers/apps/components/stateful_set_test.go
+++ /dev/null
@@ -1,181 +0,0 @@
-/*
-Copyright (C) 2022-2023 ApeCloud Co., Ltd
-
-This file is part of KubeBlocks project
-
-This program is free software: you can redistribute it and/or modify
-it under the terms of the GNU Affero General Public License as published by
-the Free Software Foundation, either version 3 of the License, or
-(at your option) any later version.
-
-This program is distributed in the hope that it will be useful
-but WITHOUT ANY WARRANTY; without even the implied warranty of
-MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
-GNU Affero General Public License for more details.
-
-You should have received a copy of the GNU Affero General Public License
-along with this program. If not, see .
-*/
-
-package components
-
-import (
- "fmt"
- "time"
-
- . "github.com/onsi/ginkgo/v2"
- . "github.com/onsi/gomega"
-
- appsv1 "k8s.io/api/apps/v1"
- corev1 "k8s.io/api/core/v1"
- metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
- "sigs.k8s.io/controller-runtime/pkg/client"
-
- appsv1alpha1 "github.com/apecloud/kubeblocks/apis/apps/v1alpha1"
- "github.com/apecloud/kubeblocks/internal/constant"
- intctrlutil "github.com/apecloud/kubeblocks/internal/generics"
- testapps "github.com/apecloud/kubeblocks/internal/testutil/apps"
- testk8s "github.com/apecloud/kubeblocks/internal/testutil/k8s"
-)
-
-var _ = Describe("Stateful Component", func() {
- var (
- randomStr = testCtx.GetRandomStr()
- clusterDefName = "mysql1-clusterdef-" + randomStr
- clusterVersionName = "mysql1-clusterversion-" + randomStr
- clusterName = "mysql1-" + randomStr
- )
- const (
- defaultMinReadySeconds = 10
- statefulCompDefRef = "stateful"
- statefulCompName = "stateful"
- )
- cleanAll := func() {
- // must wait till resources deleted and no longer existed before the testcases start,
- // otherwise if later it needs to create some new resource objects with the same name,
- // in race conditions, it will find the existence of old objects, resulting failure to
- // create the new objects.
- By("clean resources")
- // delete cluster(and all dependent sub-resources), clusterversion and clusterdef
- testapps.ClearClusterResources(&testCtx)
-
- // clear rest resources
- inNS := client.InNamespace(testCtx.DefaultNamespace)
- ml := client.HasLabels{testCtx.TestObjLabelKey}
- // namespaced resources
- testapps.ClearResources(&testCtx, intctrlutil.StatefulSetSignature, inNS, ml)
- testapps.ClearResources(&testCtx, intctrlutil.PodSignature, inNS, ml, client.GracePeriodSeconds(0))
- }
-
- BeforeEach(cleanAll)
-
- AfterEach(cleanAll)
-
- Context("Stateful Component test", func() {
- It("Stateful Component test", func() {
- By(" init cluster, statefulSet, pods")
- clusterDef, _, cluster := testapps.InitConsensusMysql(&testCtx, clusterDefName,
- clusterVersionName, clusterName, statefulCompDefRef, statefulCompName)
- _ = testapps.MockConsensusComponentStatefulSet(&testCtx, clusterName, statefulCompName)
- stsList := &appsv1.StatefulSetList{}
- Eventually(func() bool {
- _ = k8sClient.List(ctx, stsList, client.InNamespace(testCtx.DefaultNamespace), client.MatchingLabels{
- constant.AppInstanceLabelKey: clusterName,
- constant.KBAppComponentLabelKey: statefulCompName,
- }, client.Limit(1))
- return len(stsList.Items) > 0
- }).Should(BeTrue())
-
- By("test pods number of sts is 0")
- sts := &stsList.Items[0]
- clusterComponent := cluster.Spec.GetComponentByName(statefulCompName)
- componentDef := clusterDef.GetComponentDefByName(clusterComponent.ComponentDefRef)
- stateful := newStateful(k8sClient, cluster, clusterComponent, *componentDef)
- phase, _, _ := stateful.GetPhaseWhenPodsNotReady(ctx, statefulCompName, false)
- Expect(phase == appsv1alpha1.FailedClusterCompPhase).Should(BeTrue())
-
- By("test pods are not ready")
- updateRevision := fmt.Sprintf("%s-%s-%s", clusterName, statefulCompName, "6fdd48d9cd")
- Expect(testapps.ChangeObjStatus(&testCtx, sts, func() {
- availableReplicas := *sts.Spec.Replicas - 1
- sts.Status.AvailableReplicas = availableReplicas
- sts.Status.ReadyReplicas = availableReplicas
- sts.Status.Replicas = availableReplicas
- sts.Status.ObservedGeneration = 1
- sts.Status.UpdateRevision = updateRevision
- })).Should(Succeed())
- podsReady, _ := stateful.PodsReady(ctx, sts)
- Expect(podsReady).Should(BeFalse())
-
- By("create pods of sts")
- podList := testapps.MockConsensusComponentPods(&testCtx, sts, clusterName, statefulCompName)
-
- By("test stateful component is abnormal")
- pod := podList[0]
- // mock pod is not ready
- Expect(testapps.ChangeObjStatus(&testCtx, pod, func() {
- pod.Status.Conditions = []corev1.PodCondition{}
- })).Should(Succeed())
- // mock pod scheduled failure
- // testk8s.UpdatePodStatusScheduleFailed(ctx, testCtx, pod.Name, pod.Namespace)
- Eventually(testapps.CheckObj(&testCtx, client.ObjectKeyFromObject(sts), func(g Gomega, tmpSts *appsv1.StatefulSet) {
- g.Expect(tmpSts.Status.AvailableReplicas == *sts.Spec.Replicas-1).Should(BeTrue())
- })).Should(Succeed())
-
- By("should return empty string if pod of component is only not ready when component is not up running")
- phase, _, _ = stateful.GetPhaseWhenPodsNotReady(ctx, statefulCompName, false)
- Expect(string(phase)).Should(Equal(""))
-
- By("expect component phase is Failed when pod of component is not ready and component is up running")
- phase, _, _ = stateful.GetPhaseWhenPodsNotReady(ctx, statefulCompName, true)
- Expect(phase).Should(Equal(appsv1alpha1.AbnormalClusterCompPhase))
-
- By("expect component phase is Abnormal when pod of component is failed")
- testk8s.UpdatePodStatusScheduleFailed(ctx, testCtx, pod.Name, pod.Namespace)
- phase, _, _ = stateful.GetPhaseWhenPodsNotReady(ctx, statefulCompName, false)
- Expect(phase).Should(Equal(appsv1alpha1.AbnormalClusterCompPhase))
-
- By("not ready pod is not controlled by latest revision, should return empty string")
- // mock pod is not controlled by latest revision
- Expect(testapps.ChangeObj(&testCtx, pod, func(lpod *corev1.Pod) {
- lpod.Labels[appsv1.ControllerRevisionHashLabelKey] = fmt.Sprintf("%s-%s-%s", clusterName, statefulCompName, "5wdsd8d9fs")
- })).Should(Succeed())
- phase, _, _ = stateful.GetPhaseWhenPodsNotReady(ctx, statefulCompName, false)
- Expect(string(phase)).Should(Equal(""))
- // reset updateRevision
- Expect(testapps.ChangeObj(&testCtx, pod, func(lpod *corev1.Pod) {
- lpod.Labels[appsv1.ControllerRevisionHashLabelKey] = updateRevision
- })).Should(Succeed())
-
- By("test pod is available")
- lastTransTime := metav1.NewTime(time.Now().Add(-1 * (defaultMinReadySeconds + 1) * time.Second))
- testk8s.MockPodAvailable(pod, lastTransTime)
- Expect(stateful.PodIsAvailable(pod, defaultMinReadySeconds)).Should(BeTrue())
-
- By("test pods are ready")
- // mock sts is ready
- testk8s.MockStatefulSetReady(sts)
- podsReady, _ = stateful.PodsReady(ctx, sts)
- Expect(podsReady).Should(BeTrue())
-
- By("test component.replicas is inconsistent with sts.spec.replicas")
- oldReplicas := clusterComponent.Replicas
- replicas := int32(4)
- clusterComponent.Replicas = replicas
- isRunning, _ := stateful.IsRunning(ctx, sts)
- Expect(isRunning).Should(BeFalse())
- // reset replicas
- clusterComponent.Replicas = oldReplicas
-
- By("test component is running")
- isRunning, _ = stateful.IsRunning(ctx, sts)
- Expect(isRunning).Should(BeTrue())
-
- // TODO(refactor): probe timed-out pod
- // By("test handle probe timed out")
- // requeue, _ := stateful.HandleProbeTimeoutWhenPodsReady(ctx, nil)
- // Expect(requeue == false).Should(BeTrue())
- })
- })
-
-})
diff --git a/controllers/apps/components/stateless.go b/controllers/apps/components/stateless.go
deleted file mode 100644
index 4b0492363ac..00000000000
--- a/controllers/apps/components/stateless.go
+++ /dev/null
@@ -1,304 +0,0 @@
-/*
-Copyright (C) 2022-2023 ApeCloud Co., Ltd
-
-This file is part of KubeBlocks project
-
-This program is free software: you can redistribute it and/or modify
-it under the terms of the GNU Affero General Public License as published by
-the Free Software Foundation, either version 3 of the License, or
-(at your option) any later version.
-
-This program is distributed in the hope that it will be useful
-but WITHOUT ANY WARRANTY; without even the implied warranty of
-MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
-GNU Affero General Public License for more details.
-
-You should have received a copy of the GNU Affero General Public License
-along with this program. If not, see .
-*/
-
-package components
-
-import (
- "fmt"
- "reflect"
-
- appsv1 "k8s.io/api/apps/v1"
- corev1 "k8s.io/api/core/v1"
- "k8s.io/client-go/tools/record"
- "sigs.k8s.io/controller-runtime/pkg/client"
-
- appsv1alpha1 "github.com/apecloud/kubeblocks/apis/apps/v1alpha1"
- "github.com/apecloud/kubeblocks/internal/controller/component"
- "github.com/apecloud/kubeblocks/internal/controller/graph"
- ictrltypes "github.com/apecloud/kubeblocks/internal/controller/types"
- intctrlutil "github.com/apecloud/kubeblocks/internal/controllerutil"
-)
-
-func newStatelessComponent(cli client.Client,
- recorder record.EventRecorder,
- cluster *appsv1alpha1.Cluster,
- clusterVersion *appsv1alpha1.ClusterVersion,
- synthesizedComponent *component.SynthesizedComponent,
- dag *graph.DAG) *statelessComponent {
- comp := &statelessComponent{
- componentBase: componentBase{
- Client: cli,
- Recorder: recorder,
- Cluster: cluster,
- ClusterVersion: clusterVersion,
- Component: synthesizedComponent,
- ComponentSet: &stateless{
- componentSetBase: componentSetBase{
- Cli: cli,
- Cluster: cluster,
- SynthesizedComponent: synthesizedComponent,
- ComponentSpec: nil,
- ComponentDef: nil,
- },
- },
- Dag: dag,
- WorkloadVertex: nil,
- },
- }
- return comp
-}
-
-type statelessComponent struct {
- componentBase
- // runningWorkload can be nil, and the replicas of workload can be nil (zero)
- runningWorkload *appsv1.Deployment
-}
-
-var _ Component = &statelessComponent{}
-
-func (c *statelessComponent) newBuilder(reqCtx intctrlutil.RequestCtx, cli client.Client,
- action *ictrltypes.LifecycleAction) componentWorkloadBuilder {
- builder := &statelessComponentWorkloadBuilder{
- componentWorkloadBuilderBase: componentWorkloadBuilderBase{
- ReqCtx: reqCtx,
- Client: cli,
- Comp: c,
- DefaultAction: action,
- Error: nil,
- EnvConfig: nil,
- Workload: nil,
- },
- }
- builder.ConcreteBuilder = builder
- return builder
-}
-
-func (c *statelessComponent) init(reqCtx intctrlutil.RequestCtx, cli client.Client, builder componentWorkloadBuilder, load bool) error {
- var err error
- if builder != nil {
- if err = builder.BuildEnv().
- BuildWorkload().
- BuildPDB().
- BuildHeadlessService().
- BuildConfig().
- BuildTLSVolume().
- BuildVolumeMount().
- BuildService().
- BuildTLSCert().
- Complete(); err != nil {
- return err
- }
- }
- if load {
- c.runningWorkload, err = c.loadRunningWorkload(reqCtx, cli)
- if err != nil {
- return err
- }
- }
- return nil
-}
-
-func (c *statelessComponent) loadRunningWorkload(reqCtx intctrlutil.RequestCtx, cli client.Client) (*appsv1.Deployment, error) {
- deployList, err := listDeployOwnedByComponent(reqCtx.Ctx, cli, c.GetNamespace(), c.GetMatchingLabels())
- if err != nil {
- return nil, err
- }
- cnt := len(deployList)
- if cnt == 1 {
- return deployList[0], nil
- }
- if cnt == 0 {
- return nil, nil
- } else {
- return nil, fmt.Errorf("more than one workloads found for the stateless component, cluster: %s, component: %s, cnt: %d",
- c.GetClusterName(), c.GetName(), cnt)
- }
-}
-
-func (c *statelessComponent) GetWorkloadType() appsv1alpha1.WorkloadType {
- return appsv1alpha1.Stateless
-}
-
-func (c *statelessComponent) GetBuiltObjects(reqCtx intctrlutil.RequestCtx, cli client.Client) ([]client.Object, error) {
- dag := c.Dag
- defer func() {
- c.Dag = dag
- }()
-
- c.Dag = graph.NewDAG()
- if err := c.init(intctrlutil.RequestCtx{}, nil, c.newBuilder(reqCtx, cli, ictrltypes.ActionCreatePtr()), false); err != nil {
- return nil, err
- }
-
- objs := make([]client.Object, 0)
- for _, v := range c.Dag.Vertices() {
- if vv, ok := v.(*ictrltypes.LifecycleVertex); ok {
- objs = append(objs, vv.Obj)
- }
- }
- return objs, nil
-}
-
-func (c *statelessComponent) Create(reqCtx intctrlutil.RequestCtx, cli client.Client) error {
- if err := c.init(reqCtx, cli, c.newBuilder(reqCtx, cli, ictrltypes.ActionCreatePtr()), false); err != nil {
- return err
- }
-
- if err := c.ValidateObjectsAction(); err != nil {
- return err
- }
-
- c.SetStatusPhase(appsv1alpha1.CreatingClusterCompPhase, nil, "Create a new component")
-
- return nil
-}
-
-func (c *statelessComponent) Delete(reqCtx intctrlutil.RequestCtx, cli client.Client) error {
- // TODO(impl): delete component owned resources
- return nil
-}
-
-func (c *statelessComponent) Update(reqCtx intctrlutil.RequestCtx, cli client.Client) error {
- if err := c.init(reqCtx, cli, c.newBuilder(reqCtx, cli, nil), true); err != nil {
- return err
- }
-
- if c.runningWorkload != nil {
- if err := c.Restart(reqCtx, cli); err != nil {
- return err
- }
-
- // cluster.spec.componentSpecs[*].volumeClaimTemplates[*].spec.resources.requests[corev1.ResourceStorage]
- if err := c.ExpandVolume(reqCtx, cli); err != nil {
- return err
- }
-
- // cluster.spec.componentSpecs[*].replicas
- if err := c.HorizontalScale(reqCtx, cli); err != nil {
- return err
- }
- }
-
- if err := c.updateUnderlyingResources(reqCtx, cli, c.runningWorkload); err != nil {
- return err
- }
-
- return c.ResolveObjectsAction(reqCtx, cli)
-}
-
-func (c *statelessComponent) Status(reqCtx intctrlutil.RequestCtx, cli client.Client) error {
- if err := c.init(reqCtx, cli, c.newBuilder(reqCtx, cli, ictrltypes.ActionNoopPtr()), true); err != nil {
- return err
- }
- if c.runningWorkload == nil {
- return nil
- }
-
- // patch the current componentSpec workload's custom labels
- if err := updateCustomLabelToPods(reqCtx.Ctx, cli, c.Cluster, c.Component, c.Dag); err != nil {
- reqCtx.Event(c.Cluster, corev1.EventTypeWarning, "Component Workload Controller PatchWorkloadCustomLabelFailed", err.Error())
- return err
- }
-
- return c.componentBase.StatusWorkload(reqCtx, cli, c.runningWorkload, nil)
-}
-
-func (c *statelessComponent) ExpandVolume(reqCtx intctrlutil.RequestCtx, cli client.Client) error {
- return nil
-}
-
-func (c *statelessComponent) HorizontalScale(reqCtx intctrlutil.RequestCtx, cli client.Client) error {
- if c.runningWorkload.Spec.Replicas == nil && c.Component.Replicas > 0 {
- reqCtx.Recorder.Eventf(c.Cluster,
- corev1.EventTypeNormal,
- "HorizontalScale",
- "start horizontal scale component %s of cluster %s from %d to %d",
- c.GetName(), c.GetClusterName(), 0, c.Component.Replicas)
- } else if c.runningWorkload.Spec.Replicas != nil && *c.runningWorkload.Spec.Replicas != c.Component.Replicas {
- reqCtx.Recorder.Eventf(c.Cluster,
- corev1.EventTypeNormal,
- "HorizontalScale",
- "start horizontal scale component %s of cluster %s from %d to %d",
- c.GetName(), c.GetClusterName(), *c.runningWorkload.Spec.Replicas, c.Component.Replicas)
- }
- return nil
-}
-
-func (c *statelessComponent) Restart(reqCtx intctrlutil.RequestCtx, cli client.Client) error {
- return restartPod(&c.runningWorkload.Spec.Template)
-}
-
-func (c *statelessComponent) Reconfigure(reqCtx intctrlutil.RequestCtx, cli client.Client) error {
- return nil // TODO(impl)
-}
-
-func (c *statelessComponent) updateUnderlyingResources(reqCtx intctrlutil.RequestCtx, cli client.Client, deployObj *appsv1.Deployment) error {
- if deployObj == nil {
- c.createWorkload()
- } else {
- c.updateWorkload(deployObj)
- }
- if err := c.UpdatePDB(reqCtx, cli); err != nil {
- return err
- }
- if err := c.UpdateService(reqCtx, cli); err != nil {
- return err
- }
- // update KB___ env needed by pod to obtain hostname.
- c.updatePodEnvConfig()
- return nil
-}
-
-func (c *statelessComponent) createWorkload() {
- deployProto := c.WorkloadVertex.Obj.(*appsv1.Deployment)
- c.WorkloadVertex.Obj = deployProto
- c.WorkloadVertex.Action = ictrltypes.ActionCreatePtr()
- c.SetStatusPhase(appsv1alpha1.UpdatingClusterCompPhase, nil, "Component workload created")
-}
-
-func (c *statelessComponent) updateWorkload(deployObj *appsv1.Deployment) {
- deployObjCopy := deployObj.DeepCopy()
- deployProto := c.WorkloadVertex.Obj.(*appsv1.Deployment)
-
- mergeAnnotations(deployObj.Spec.Template.Annotations, &deployProto.Spec.Template.Annotations)
- buildWorkLoadAnnotations(deployObjCopy, c.Cluster)
- deployObjCopy.Spec = deployProto.Spec
-
- resolvePodSpecDefaultFields(deployObj.Spec.Template.Spec, &deployObjCopy.Spec.Template.Spec)
-
- delayUpdatePodSpecSystemFields(deployObj.Spec.Template.Spec, &deployObjCopy.Spec.Template.Spec)
-
- if !reflect.DeepEqual(&deployObj.Spec, &deployObjCopy.Spec) {
- updatePodSpecSystemFields(&deployObjCopy.Spec.Template.Spec)
- c.WorkloadVertex.Obj = deployObjCopy
- c.WorkloadVertex.Action = ictrltypes.ActionUpdatePtr()
- c.SetStatusPhase(appsv1alpha1.UpdatingClusterCompPhase, nil, "Component workload updated")
- }
-}
-
-func (c *statelessComponent) updatePodEnvConfig() {
- for _, v := range ictrltypes.FindAll[*corev1.ConfigMap](c.Dag) {
- node := v.(*ictrltypes.LifecycleVertex)
- // TODO: need a way to reference the env config.
- envConfigName := fmt.Sprintf("%s-%s-env", c.GetClusterName(), c.GetName())
- if node.Obj.GetName() == envConfigName {
- node.Action = ictrltypes.ActionUpdatePtr()
- }
- }
-}
diff --git a/controllers/apps/components/stateless_set.go b/controllers/apps/components/stateless_set.go
deleted file mode 100644
index 3738625988b..00000000000
--- a/controllers/apps/components/stateless_set.go
+++ /dev/null
@@ -1,196 +0,0 @@
-/*
-Copyright (C) 2022-2023 ApeCloud Co., Ltd
-
-This file is part of KubeBlocks project
-
-This program is free software: you can redistribute it and/or modify
-it under the terms of the GNU Affero General Public License as published by
-the Free Software Foundation, either version 3 of the License, or
-(at your option) any later version.
-
-This program is distributed in the hope that it will be useful
-but WITHOUT ANY WARRANTY; without even the implied warranty of
-MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
-GNU Affero General Public License for more details.
-
-You should have received a copy of the GNU Affero General Public License
-along with this program. If not, see .
-*/
-
-package components
-
-import (
- "context"
- "math"
- "strings"
- "time"
-
- appsv1 "k8s.io/api/apps/v1"
- corev1 "k8s.io/api/core/v1"
- metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
- deploymentutil "k8s.io/kubectl/pkg/util/deployment"
- "k8s.io/kubectl/pkg/util/podutils"
- "sigs.k8s.io/controller-runtime/pkg/client"
-
- appsv1alpha1 "github.com/apecloud/kubeblocks/apis/apps/v1alpha1"
- "github.com/apecloud/kubeblocks/internal/constant"
- "github.com/apecloud/kubeblocks/internal/controller/graph"
- intctrlutil "github.com/apecloud/kubeblocks/internal/controllerutil"
-)
-
-// NewRSAvailableReason is added in a deployment when its newest replica set is made available
-// ie. the number of new pods that have passed readiness checks and run for at least minReadySeconds
-// is at least the minimum available pods that need to run for the deployment.
-const NewRSAvailableReason = "NewReplicaSetAvailable"
-
-type stateless struct {
- componentSetBase
-}
-
-var _ componentSet = &stateless{}
-
-func (stateless *stateless) getReplicas() int32 {
- if stateless.SynthesizedComponent != nil {
- return stateless.SynthesizedComponent.Replicas
- }
- return stateless.ComponentSpec.Replicas
-}
-
-func (stateless *stateless) IsRunning(ctx context.Context, obj client.Object) (bool, error) {
- if stateless == nil {
- return false, nil
- }
- return stateless.PodsReady(ctx, obj)
-}
-
-func (stateless *stateless) PodsReady(ctx context.Context, obj client.Object) (bool, error) {
- if stateless == nil {
- return false, nil
- }
- deploy, ok := obj.(*appsv1.Deployment)
- if !ok {
- return false, nil
- }
- targetReplicas := stateless.getReplicas()
- return deploymentIsReady(deploy, &targetReplicas), nil
-}
-
-func (stateless *stateless) PodIsAvailable(pod *corev1.Pod, minReadySeconds int32) bool {
- if stateless == nil || pod == nil {
- return false
- }
- return podutils.IsPodAvailable(pod, minReadySeconds, metav1.Time{Time: time.Now()})
-}
-
-func (stateless *stateless) GetPhaseWhenPodsReadyAndProbeTimeout(pods []*corev1.Pod) (appsv1alpha1.ClusterComponentPhase, appsv1alpha1.ComponentMessageMap) {
- return "", nil
-}
-
-// GetPhaseWhenPodsNotReady gets the component phase when the pods of component are not ready.
-func (stateless *stateless) GetPhaseWhenPodsNotReady(ctx context.Context,
- componentName string,
- originPhaseIsUpRunning bool) (appsv1alpha1.ClusterComponentPhase, appsv1alpha1.ComponentMessageMap, error) {
- deployList := &appsv1.DeploymentList{}
- podList, err := getCompRelatedObjectList(ctx, stateless.Cli, *stateless.Cluster, componentName, deployList)
- if err != nil || len(deployList.Items) == 0 {
- return "", nil, err
- }
- statusMessages := appsv1alpha1.ComponentMessageMap{}
- // if the failed pod is not controlled by the new ReplicaSetKind
- checkExistFailedPodOfNewRS := func(pod *corev1.Pod, workload metav1.Object) bool {
- d := workload.(*appsv1.Deployment)
- // if component is up running but pod is not ready, this pod should be failed.
- // for example: full disk cause readiness probe failed and serve is not available.
- // but kubelet only sets the container is not ready and pod is also Running.
- if originPhaseIsUpRunning {
- return !intctrlutil.PodIsReady(pod) && belongToNewReplicaSet(d, pod)
- }
- isFailed, _, message := IsPodFailedAndTimedOut(pod)
- existLatestRevisionFailedPod := isFailed && belongToNewReplicaSet(d, pod)
- if existLatestRevisionFailedPod {
- statusMessages.SetObjectMessage(pod.Kind, pod.Name, message)
- }
- return existLatestRevisionFailedPod
- }
- deploy := &deployList.Items[0]
- return getComponentPhaseWhenPodsNotReady(podList, deploy, stateless.getReplicas(),
- deploy.Status.AvailableReplicas, nil, checkExistFailedPodOfNewRS), statusMessages, nil
-}
-
-func (stateless *stateless) HandleRestart(context.Context, client.Object) ([]graph.Vertex, error) {
- return nil, nil
-}
-
-func (stateless *stateless) HandleRoleChange(context.Context, client.Object) ([]graph.Vertex, error) {
- return nil, nil
-}
-
-func newStateless(cli client.Client,
- cluster *appsv1alpha1.Cluster,
- spec *appsv1alpha1.ClusterComponentSpec,
- def appsv1alpha1.ClusterComponentDefinition) *stateless {
- return &stateless{
- componentSetBase: componentSetBase{
- Cli: cli,
- Cluster: cluster,
- SynthesizedComponent: nil,
- ComponentSpec: spec,
- ComponentDef: &def,
- },
- }
-}
-
-// deploymentIsReady checks deployment is ready
-func deploymentIsReady(deploy *appsv1.Deployment, targetReplicas *int32) bool {
- var (
- componentIsRunning = true
- newRSAvailable = true
- )
- if targetReplicas == nil {
- targetReplicas = deploy.Spec.Replicas
- }
-
- if hasProgressDeadline(deploy) {
- // if the deployment.Spec.ProgressDeadlineSeconds exists, we should check if the new replicaSet is available.
- // when deployment.Spec.ProgressDeadlineSeconds does not exist, the deployment controller will remove the
- // DeploymentProgressing condition.
- condition := deploymentutil.GetDeploymentCondition(deploy.Status, appsv1.DeploymentProgressing)
- if condition == nil || condition.Reason != NewRSAvailableReason || condition.Status != corev1.ConditionTrue {
- newRSAvailable = false
- }
- }
- // check if the deployment of component is updated completely and ready.
- if deploy.Status.AvailableReplicas != *targetReplicas ||
- deploy.Status.Replicas != *targetReplicas ||
- deploy.Status.ObservedGeneration != deploy.Generation ||
- deploy.Status.UpdatedReplicas != *targetReplicas ||
- !newRSAvailable {
- componentIsRunning = false
- }
- return componentIsRunning
-}
-
-// hasProgressDeadline checks if the Deployment d is expected to suffice the reason
-// "ProgressDeadlineExceeded" when the Deployment progress takes longer than expected time.
-func hasProgressDeadline(d *appsv1.Deployment) bool {
- return d.Spec.ProgressDeadlineSeconds != nil &&
- *d.Spec.ProgressDeadlineSeconds > 0 &&
- *d.Spec.ProgressDeadlineSeconds != math.MaxInt32
-}
-
-// belongToNewReplicaSet checks if the pod belongs to the new replicaSet of deployment
-func belongToNewReplicaSet(d *appsv1.Deployment, pod *corev1.Pod) bool {
- if pod == nil || d == nil {
- return false
- }
- condition := deploymentutil.GetDeploymentCondition(d.Status, appsv1.DeploymentProgressing)
- if condition == nil {
- return false
- }
- for _, v := range pod.OwnerReferences {
- if v.Kind == constant.ReplicaSetKind && strings.Contains(condition.Message, v.Name) {
- return d.Status.ObservedGeneration == d.Generation
- }
- }
- return false
-}
diff --git a/controllers/apps/components/stateless_set_test.go b/controllers/apps/components/stateless_set_test.go
deleted file mode 100644
index a4d142ed9b1..00000000000
--- a/controllers/apps/components/stateless_set_test.go
+++ /dev/null
@@ -1,173 +0,0 @@
-/*
-Copyright (C) 2022-2023 ApeCloud Co., Ltd
-
-This file is part of KubeBlocks project
-
-This program is free software: you can redistribute it and/or modify
-it under the terms of the GNU Affero General Public License as published by
-the Free Software Foundation, either version 3 of the License, or
-(at your option) any later version.
-
-This program is distributed in the hope that it will be useful
-but WITHOUT ANY WARRANTY; without even the implied warranty of
-MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
-GNU Affero General Public License for more details.
-
-You should have received a copy of the GNU Affero General Public License
-along with this program. If not, see .
-*/
-
-package components
-
-import (
- "fmt"
- "time"
-
- . "github.com/onsi/ginkgo/v2"
- . "github.com/onsi/gomega"
-
- appsv1 "k8s.io/api/apps/v1"
- corev1 "k8s.io/api/core/v1"
- metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
- "sigs.k8s.io/controller-runtime/pkg/client"
-
- appsv1alpha1 "github.com/apecloud/kubeblocks/apis/apps/v1alpha1"
- intctrlutil "github.com/apecloud/kubeblocks/internal/generics"
- testapps "github.com/apecloud/kubeblocks/internal/testutil/apps"
- testk8s "github.com/apecloud/kubeblocks/internal/testutil/k8s"
-)
-
-var _ = Describe("Stateful Component", func() {
- var (
- randomStr = testCtx.GetRandomStr()
- clusterDefName = "stateless-definition-" + randomStr
- clusterVersionName = "stateless-cluster-version-" + randomStr
- clusterName = "stateless-" + randomStr
- )
- const (
- statelessCompName = "stateless"
- statelessCompDefName = "stateless"
- defaultMinReadySeconds = 10
- )
-
- cleanAll := func() {
- // must wait till resources deleted and no longer existed before the testcases start,
- // otherwise if later it needs to create some new resource objects with the same name,
- // in race conditions, it will find the existence of old objects, resulting failure to
- // create the new objects.
- By("clean resources")
-
- // clear rest resources
- inNS := client.InNamespace(testCtx.DefaultNamespace)
- ml := client.HasLabels{testCtx.TestObjLabelKey}
- // namespaced resources
- testapps.ClearResources(&testCtx, intctrlutil.ClusterSignature, inNS, ml)
- testapps.ClearResources(&testCtx, intctrlutil.DeploymentSignature, inNS, ml)
- testapps.ClearResources(&testCtx, intctrlutil.PodSignature, inNS, ml, client.GracePeriodSeconds(0))
- }
-
- BeforeEach(cleanAll)
-
- AfterEach(cleanAll)
-
- Context("Stateless Component test", func() {
- It("Stateless Component test", func() {
- By(" init cluster, deployment")
- clusterDef := testapps.NewClusterDefFactory(clusterDefName).
- AddComponentDef(testapps.StatelessNginxComponent, statelessCompDefName).
- Create(&testCtx).GetObject()
- cluster := testapps.NewClusterFactory(testCtx.DefaultNamespace, clusterName, clusterDefName, clusterVersionName).
- AddComponent(statelessCompName, statelessCompDefName).SetReplicas(2).Create(&testCtx).GetObject()
- deploy := testapps.MockStatelessComponentDeploy(&testCtx, clusterName, statelessCompName)
- clusterComponent := cluster.Spec.GetComponentByName(statelessCompName)
- componentDef := clusterDef.GetComponentDefByName(clusterComponent.ComponentDefRef)
- statelessComponent := newStateless(k8sClient, cluster, clusterComponent, *componentDef)
- By("test pods number of deploy is 0 ")
- phase, _, _ := statelessComponent.GetPhaseWhenPodsNotReady(ctx, statelessCompName, false)
- Expect(phase == appsv1alpha1.FailedClusterCompPhase).Should(BeTrue())
-
- By("test pod is ready")
- rsName := deploy.Name + "-5847cb795c"
- pod := testapps.MockStatelessPod(&testCtx, deploy, clusterName, statelessCompName, rsName+randomStr)
- lastTransTime := metav1.NewTime(time.Now().Add(-1 * (defaultMinReadySeconds + 1) * time.Second))
- testk8s.MockPodAvailable(pod, lastTransTime)
- Expect(statelessComponent.PodIsAvailable(pod, defaultMinReadySeconds)).Should(BeTrue())
-
- By("test a part pods of deploy are not ready")
- // mock pod is not ready
- Expect(testapps.ChangeObjStatus(&testCtx, pod, func() {
- pod.Status.Conditions = []corev1.PodCondition{}
- })).Should(Succeed())
- // mock deployment is processing rs
- Expect(testapps.ChangeObjStatus(&testCtx, deploy, func() {
- deploy.Status.Conditions = []appsv1.DeploymentCondition{
- {
- Type: appsv1.DeploymentProgressing,
- Reason: "ProcessingRs",
- Status: corev1.ConditionTrue,
- Message: fmt.Sprintf(`ReplicaSet "%s" has progressing.`, rsName),
- },
- }
- deploy.Status.ObservedGeneration = 1
- })).Should(Succeed())
- Expect(testapps.ChangeObjStatus(&testCtx, deploy, func() {
- availableReplicas := *deploy.Spec.Replicas - 1
- deploy.Status.AvailableReplicas = availableReplicas
- deploy.Status.ReadyReplicas = availableReplicas
- deploy.Status.Replicas = availableReplicas
- })).Should(Succeed())
- podsReady, _ := statelessComponent.PodsReady(ctx, deploy)
- Expect(podsReady).Should(BeFalse())
- By("should return empty string if pod of component is only not ready when component is not up running")
- phase, _, _ = statelessComponent.GetPhaseWhenPodsNotReady(ctx, statelessCompName, false)
- Expect(string(phase)).Should(Equal(""))
-
- By("expect component phase is Failed when pod of component is not ready and component is up running")
- phase, _, _ = statelessComponent.GetPhaseWhenPodsNotReady(ctx, statelessCompName, true)
- Expect(phase).Should(Equal(appsv1alpha1.AbnormalClusterCompPhase))
-
- By("expect component phase is Abnormal when pod of component is failed")
- testk8s.UpdatePodStatusScheduleFailed(ctx, testCtx, pod.Name, pod.Namespace)
- phase, _, _ = statelessComponent.GetPhaseWhenPodsNotReady(ctx, statelessCompName, false)
- Expect(phase).Should(Equal(appsv1alpha1.AbnormalClusterCompPhase))
-
- By("test pods of deployment are ready")
- testk8s.MockDeploymentReady(deploy, NewRSAvailableReason, rsName)
- podsReady, _ = statelessComponent.PodsReady(ctx, deploy)
- Expect(podsReady).Should(BeTrue())
-
- By("test component.replicas is inconsistent with deployment.spec.replicas")
- oldReplicas := clusterComponent.Replicas
- replicas := int32(4)
- clusterComponent.Replicas = replicas
- isRunning, _ := statelessComponent.IsRunning(ctx, deploy)
- Expect(isRunning).Should(BeFalse())
- // reset replicas
- clusterComponent.Replicas = oldReplicas
-
- By("test component is running")
- isRunning, _ = statelessComponent.IsRunning(ctx, deploy)
- Expect(isRunning).Should(BeTrue())
-
- // TODO(refactor): probe timed-out pod
- // By("test handle probe timed out")
- // requeue, _ := statelessComponent.HandleProbeTimeoutWhenPodsReady(ctx, nil)
- // Expect(requeue == false).Should(BeTrue())
-
- By("test pod is not failed and not controlled by new ReplicaSet of deployment")
- Expect(testapps.ChangeObjStatus(&testCtx, deploy, func() {
- deploy.Status.Conditions = []appsv1.DeploymentCondition{
- {
- Type: appsv1.DeploymentProgressing,
- Reason: "ProcessingRs",
- Status: corev1.ConditionTrue,
- Message: fmt.Sprintf(`ReplicaSet "%s" has progressing.`, deploy.Name+"-584f7csdb"),
- },
- }
- })).Should(Succeed())
- phase, _, _ = statelessComponent.GetPhaseWhenPodsNotReady(ctx, statelessCompName, false)
- Expect(string(phase)).Should(Equal(""))
- })
- })
-
-})
diff --git a/controllers/apps/components/status.go b/controllers/apps/components/status.go
deleted file mode 100644
index 0eb6cc4ca94..00000000000
--- a/controllers/apps/components/status.go
+++ /dev/null
@@ -1,63 +0,0 @@
-/*
-Copyright (C) 2022-2023 ApeCloud Co., Ltd
-
-This file is part of KubeBlocks project
-
-This program is free software: you can redistribute it and/or modify
-it under the terms of the GNU Affero General Public License as published by
-the Free Software Foundation, either version 3 of the License, or
-(at your option) any later version.
-
-This program is distributed in the hope that it will be useful
-but WITHOUT ANY WARRANTY; without even the implied warranty of
-MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
-GNU Affero General Public License for more details.
-
-You should have received a copy of the GNU Affero General Public License
-along with this program. If not, see .
-*/
-
-package components
-
-import (
- "sort"
-
- "golang.org/x/exp/maps"
-
- appsv1alpha1 "github.com/apecloud/kubeblocks/apis/apps/v1alpha1"
-)
-
-var componentPhasePriority = map[appsv1alpha1.ClusterComponentPhase]int{
- appsv1alpha1.FailedClusterCompPhase: 1,
- appsv1alpha1.AbnormalClusterCompPhase: 2,
- appsv1alpha1.UpdatingClusterCompPhase: 3,
- appsv1alpha1.StoppedClusterCompPhase: 4,
- appsv1alpha1.RunningClusterCompPhase: 5,
- appsv1alpha1.CreatingClusterCompPhase: 6,
-}
-
-type statusReconciliationTxn struct {
- proposals map[appsv1alpha1.ClusterComponentPhase]func()
-}
-
-func (t *statusReconciliationTxn) propose(phase appsv1alpha1.ClusterComponentPhase, mutator func()) {
- if t.proposals == nil {
- t.proposals = make(map[appsv1alpha1.ClusterComponentPhase]func())
- }
- if _, ok := t.proposals[phase]; ok {
- return // keep first
- }
- t.proposals[phase] = mutator
-}
-
-func (t *statusReconciliationTxn) commit() error {
- if len(t.proposals) == 0 {
- return nil
- }
- phases := maps.Keys(t.proposals)
- sort.Slice(phases, func(i, j int) bool {
- return componentPhasePriority[phases[i]] < componentPhasePriority[phases[j]]
- })
- t.proposals[phases[0]]()
- return nil
-}
diff --git a/controllers/apps/components/status_test.go b/controllers/apps/components/status_test.go
deleted file mode 100644
index 2a8ed5ae4cc..00000000000
--- a/controllers/apps/components/status_test.go
+++ /dev/null
@@ -1,520 +0,0 @@
-/*
-Copyright (C) 2022-2023 ApeCloud Co., Ltd
-
-This file is part of KubeBlocks project
-
-This program is free software: you can redistribute it and/or modify
-it under the terms of the GNU Affero General Public License as published by
-the Free Software Foundation, either version 3 of the License, or
-(at your option) any later version.
-
-This program is distributed in the hope that it will be useful
-but WITHOUT ANY WARRANTY; without even the implied warranty of
-MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
-GNU Affero General Public License for more details.
-
-You should have received a copy of the GNU Affero General Public License
-along with this program. If not, see .
-*/
-
-package components
-
-import (
- "fmt"
- "strconv"
- "time"
-
- . "github.com/onsi/ginkgo/v2"
- . "github.com/onsi/gomega"
-
- appsv1 "k8s.io/api/apps/v1"
- corev1 "k8s.io/api/core/v1"
- metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
- "sigs.k8s.io/controller-runtime/pkg/client"
- "sigs.k8s.io/controller-runtime/pkg/log"
-
- appsv1alpha1 "github.com/apecloud/kubeblocks/apis/apps/v1alpha1"
- workloads "github.com/apecloud/kubeblocks/apis/workloads/v1alpha1"
- "github.com/apecloud/kubeblocks/internal/constant"
- "github.com/apecloud/kubeblocks/internal/controller/graph"
- intctrlutil "github.com/apecloud/kubeblocks/internal/controllerutil"
- "github.com/apecloud/kubeblocks/internal/generics"
- testapps "github.com/apecloud/kubeblocks/internal/testutil/apps"
- testk8s "github.com/apecloud/kubeblocks/internal/testutil/k8s"
-)
-
-var _ = Describe("ComponentStatusSynchronizer", func() {
- const (
- compName = "comp"
- compDefName = "comp"
- )
-
- var (
- clusterDefName = "test-clusterdef"
- clusterVersionName = "test-clusterversion"
- clusterName = "test-cluster"
- controllerRevision = fmt.Sprintf("%s-%s-%s", clusterName, compName, "6fdd48d9cd1")
-
- clusterDef *appsv1alpha1.ClusterDefinition
- cluster *appsv1alpha1.Cluster
- component Component
- rsm *workloads.ReplicatedStateMachine
- reqCtx *intctrlutil.RequestCtx
- dag *graph.DAG
- err error
- )
-
- cleanAll := func() {
- // must wait till resources deleted and no longer existed before the testcases start,
- // otherwise if later it needs to create some new resource objects with the same name,
- // in race conditions, it will find the existence of old objects, resulting failure to
- // create the new objects.
- By("clean resources")
-
- // clear rest resources
- inNS := client.InNamespace(testCtx.DefaultNamespace)
- ml := client.HasLabels{testCtx.TestObjLabelKey}
-
- // non-namespaced resources
- testapps.ClearResources(&testCtx, generics.ClusterDefinitionSignature, inNS, ml)
-
- // namespaced resources
- testapps.ClearResources(&testCtx, generics.ClusterSignature, inNS, ml)
- testapps.ClearResources(&testCtx, generics.StatefulSetSignature, inNS, ml)
- testapps.ClearResources(&testCtx, generics.DeploymentSignature, inNS, ml)
- if intctrlutil.IsRSMEnabled() {
- testapps.ClearResources(&testCtx, generics.RSMSignature, inNS, ml)
- }
-
- testapps.ClearResources(&testCtx, generics.PodSignature, inNS, ml, client.GracePeriodSeconds(0))
- }
-
- BeforeEach(cleanAll)
-
- AfterEach(cleanAll)
-
- Context("with stateless component", func() {
- BeforeEach(func() {
- clusterDef = testapps.NewClusterDefFactory(clusterDefName).
- AddComponentDef(testapps.StatelessNginxComponent, compDefName).
- GetObject()
-
- cluster = testapps.NewClusterFactory(testCtx.DefaultNamespace, clusterName, clusterDefName, clusterVersionName).
- AddComponent(compName, compDefName).
- SetReplicas(1).
- GetObject()
-
- reqCtx = &intctrlutil.RequestCtx{
- Ctx: ctx,
- Log: log.FromContext(ctx).WithValues("cluster", clusterDef.Name),
- }
- dag = graph.NewDAG()
- component, err = NewComponent(*reqCtx, testCtx.Cli, clusterDef, nil, cluster, compName, dag)
- Expect(err).Should(Succeed())
- Expect(component).ShouldNot(BeNil())
- })
-
- It("should not change component if no deployment or pod exists", func() {
- Expect(component.Status(*reqCtx, testCtx.Cli)).Should(Succeed())
- Expect(cluster.Status.Components[compName].Phase).Should(BeEmpty())
- })
-
- Context("and with mocked deployment & pod", func() {
- var (
- deployment *appsv1.Deployment
- pod *corev1.Pod
- )
-
- BeforeEach(func() {
- deploymentName := clusterName + "-" + compName
- deployment = testapps.NewDeploymentFactory(testCtx.DefaultNamespace, deploymentName, clusterName, compName).
- AddAnnotations(constant.KubeBlocksGenerationKey, strconv.FormatInt(cluster.Generation, 10)).
- SetMinReadySeconds(int32(10)).
- SetReplicas(int32(1)).
- AddContainer(corev1.Container{Name: testapps.DefaultNginxContainerName, Image: testapps.NginxImage}).
- Create(&testCtx).GetObject()
- if intctrlutil.IsRSMEnabled() {
- rsm = testapps.NewRSMFactory(testCtx.DefaultNamespace, deploymentName, clusterName, compName).
- AddAnnotations(constant.KubeBlocksGenerationKey, strconv.FormatInt(cluster.Generation, 10)).
- SetReplicas(int32(1)).
- AddContainer(corev1.Container{Name: testapps.DefaultNginxContainerName, Image: testapps.NginxImage}).
- Create(&testCtx).GetObject()
- sts := testapps.NewStatefulSetFactory(testCtx.DefaultNamespace, deploymentName, clusterName, compName).
- AddAnnotations(constant.KubeBlocksGenerationKey, strconv.FormatInt(cluster.Generation, 10)).
- SetReplicas(int32(1)).
- AddContainer(corev1.Container{Name: testapps.DefaultNginxContainerName, Image: testapps.NginxImage}).
- Create(&testCtx).GetObject()
- Expect(testapps.ChangeObjStatus(&testCtx, sts, func() {
- sts.Status.ObservedGeneration = sts.Generation
- sts.Status.UpdateRevision = controllerRevision
- })).Should(Succeed())
- Expect(testapps.ChangeObjStatus(&testCtx, rsm, func() {
- rsm.Status.InitReplicas = *rsm.Spec.Replicas
- rsm.Status.Replicas = *rsm.Spec.Replicas
- rsm.Status.UpdateRevision = controllerRevision
- rsm.Status.ObservedGeneration = rsm.Generation
- rsm.Status.CurrentGeneration = rsm.Generation
- })).Should(Succeed())
- }
-
- podName := fmt.Sprintf("%s-%s-%s", clusterName, compName, testCtx.GetRandomStr())
- if intctrlutil.IsRSMEnabled() {
- podName = rsm.Name + "-0"
- }
- pod = testapps.NewPodFactory(testCtx.DefaultNamespace, podName).
- SetOwnerReferences("apps/v1", constant.DeploymentKind, deployment).
- AddAppInstanceLabel(clusterName).
- AddAppComponentLabel(compName).
- AddAppManangedByLabel().
- AddControllerRevisionHashLabel(controllerRevision).
- AddContainer(corev1.Container{Name: testapps.DefaultNginxContainerName, Image: testapps.NginxImage}).
- Create(&testCtx).GetObject()
- })
-
- It("should set component status to failed if container is not ready and have error message", func() {
- Expect(mockContainerError(pod)).Should(Succeed())
-
- Expect(component.Status(*reqCtx, testCtx.Cli)).Should(Succeed())
- Expect(cluster.Status.Components[compName].Phase).Should(Equal(appsv1alpha1.FailedClusterCompPhase))
- })
-
- It("should set component status to running if container is ready", func() {
- Expect(testapps.ChangeObjStatus(&testCtx, deployment, func() {
- testk8s.MockDeploymentReady(deployment, NewRSAvailableReason, deployment.Name)
- })).Should(Succeed())
- if intctrlutil.IsRSMEnabled() {
- Expect(testapps.ChangeObjStatus(&testCtx, rsm, func() {
- testk8s.MockRSMReady(rsm, pod)
- })).Should(Succeed())
- }
-
- Expect(component.Status(*reqCtx, testCtx.Cli)).Should(Succeed())
- Expect(cluster.Status.Components[compName].Phase).Should(Equal(appsv1alpha1.RunningClusterCompPhase))
- })
- })
- })
-
- Context("with statefulset component", func() {
- BeforeEach(func() {
- clusterDef = testapps.NewClusterDefFactory(clusterDefName).
- AddComponentDef(testapps.StatefulMySQLComponent, compDefName).
- GetObject()
-
- cluster = testapps.NewClusterFactory(testCtx.DefaultNamespace, clusterName, clusterDefName, clusterVersionName).
- AddComponent(compName, compDefName).
- SetReplicas(int32(3)).
- GetObject()
-
- reqCtx = &intctrlutil.RequestCtx{
- Ctx: ctx,
- Log: log.FromContext(ctx).WithValues("cluster", clusterDef.Name),
- }
- dag = graph.NewDAG()
- component, err = NewComponent(*reqCtx, testCtx.Cli, clusterDef, nil, cluster, compName, dag)
- Expect(err).Should(Succeed())
- Expect(component).ShouldNot(BeNil())
- })
-
- It("should not change component if no statefulset or pod exists", func() {
- Expect(component.Status(*reqCtx, testCtx.Cli)).Should(Succeed())
- Expect(cluster.Status.Components[compName].Phase).Should(BeEmpty())
- })
-
- Context("and with mocked statefulset & pod", func() {
- var (
- statefulset *appsv1.StatefulSet
- pods []*corev1.Pod
- )
-
- BeforeEach(func() {
- stsName := clusterName + "-" + compName
- statefulset = testapps.NewStatefulSetFactory(testCtx.DefaultNamespace, stsName, clusterName, compName).
- AddAnnotations(constant.KubeBlocksGenerationKey, strconv.FormatInt(cluster.Generation, 10)).
- SetReplicas(int32(3)).
- AddContainer(corev1.Container{Name: testapps.DefaultMySQLContainerName, Image: testapps.ApeCloudMySQLImage}).
- Create(&testCtx).GetObject()
- // init statefulset status
- testk8s.InitStatefulSetStatus(testCtx, statefulset, controllerRevision)
- for i := 0; i < 3; i++ {
- podName := fmt.Sprintf("%s-%s-%d", clusterName, compName, i)
- pod := testapps.NewPodFactory(testCtx.DefaultNamespace, podName).
- SetOwnerReferences("apps/v1", constant.StatefulSetKind, statefulset).
- AddAppInstanceLabel(clusterName).
- AddAppComponentLabel(compName).
- AddAppManangedByLabel().
- AddControllerRevisionHashLabel(controllerRevision).
- AddContainer(corev1.Container{Name: testapps.DefaultMySQLContainerName, Image: testapps.ApeCloudMySQLImage}).
- Create(&testCtx).GetObject()
- Expect(testapps.ChangeObjStatus(&testCtx, pod, func() {
- pod.Status.Conditions = []corev1.PodCondition{{
- Type: corev1.PodReady,
- Status: corev1.ConditionTrue,
- }}
- })).Should(Succeed())
- pods = append(pods, pod)
- }
- if intctrlutil.IsRSMEnabled() {
- rsm = testapps.NewRSMFactory(testCtx.DefaultNamespace, stsName, clusterName, compName).
- AddAnnotations(constant.KubeBlocksGenerationKey, strconv.FormatInt(cluster.Generation, 10)).
- SetReplicas(int32(3)).
- AddContainer(corev1.Container{Name: testapps.DefaultMySQLContainerName, Image: testapps.ApeCloudMySQLImage}).
- Create(&testCtx).GetObject()
- // init rsm status
- testk8s.InitRSMStatus(testCtx, rsm, controllerRevision)
- }
- })
-
- It("should set component status to failed if container is not ready and have error message", func() {
- Expect(mockContainerError(pods[0])).Should(Succeed())
- Expect(mockContainerError(pods[1])).Should(Succeed())
-
- Expect(component.Status(*reqCtx, testCtx.Cli)).Should(Succeed())
- Expect(cluster.Status.Components[compName].Phase).Should(Equal(appsv1alpha1.FailedClusterCompPhase))
- })
-
- It("should set component status to running if container is ready", func() {
- Expect(testapps.ChangeObjStatus(&testCtx, statefulset, func() {
- testk8s.MockStatefulSetReady(statefulset)
- })).Should(Succeed())
- if intctrlutil.IsRSMEnabled() {
- Expect(testapps.ChangeObjStatus(&testCtx, rsm, func() {
- testk8s.MockRSMReady(rsm, pods...)
- })).Should(Succeed())
- }
-
- Expect(component.Status(*reqCtx, testCtx.Cli)).Should(Succeed())
- Expect(cluster.Status.Components[compName].Phase).Should(Equal(appsv1alpha1.RunningClusterCompPhase))
- })
- })
- })
-
- Context("with consensusset component", func() {
- BeforeEach(func() {
- clusterDef = testapps.NewClusterDefFactory(clusterDefName).
- AddComponentDef(testapps.ConsensusMySQLComponent, compDefName).
- Create(&testCtx).GetObject()
-
- cluster = testapps.NewClusterFactory(testCtx.DefaultNamespace, clusterName, clusterDefName, clusterVersionName).
- AddComponent(compName, compDefName).
- SetReplicas(int32(3)).
- Create(&testCtx).GetObject()
-
- reqCtx = &intctrlutil.RequestCtx{
- Ctx: ctx,
- Log: log.FromContext(ctx).WithValues("cluster", clusterDef.Name),
- }
- dag = graph.NewDAG()
- component, err = NewComponent(*reqCtx, testCtx.Cli, clusterDef, nil, cluster, compName, dag)
- Expect(err).Should(Succeed())
- Expect(component).ShouldNot(BeNil())
- })
-
- It("should not change component if no statefulset or pod exists", func() {
- Expect(component.Status(*reqCtx, testCtx.Cli)).Should(Succeed())
- Expect(cluster.Status.Components[compName].Phase).Should(BeEmpty())
- })
-
- Context("and with mocked statefulset & pod", func() {
- var (
- statefulset *appsv1.StatefulSet
- pods []*corev1.Pod
- )
-
- BeforeEach(func() {
- stsName := clusterName + "-" + compName
- statefulset = testapps.NewStatefulSetFactory(testCtx.DefaultNamespace, stsName, clusterName, compName).
- AddAnnotations(constant.KubeBlocksGenerationKey, strconv.FormatInt(cluster.Generation, 10)).
- SetReplicas(int32(3)).
- AddContainer(corev1.Container{Name: testapps.DefaultMySQLContainerName, Image: testapps.ApeCloudMySQLImage}).
- Create(&testCtx).GetObject()
- testk8s.InitStatefulSetStatus(testCtx, statefulset, controllerRevision)
- for i := 0; i < 3; i++ {
- podName := fmt.Sprintf("%s-%s-%d", clusterName, compName, i)
- pod := testapps.NewPodFactory(testCtx.DefaultNamespace, podName).
- SetOwnerReferences("apps/v1", constant.StatefulSetKind, statefulset).
- AddAppInstanceLabel(clusterName).
- AddAppComponentLabel(compName).
- AddAppManangedByLabel().
- AddControllerRevisionHashLabel(controllerRevision).
- AddContainer(corev1.Container{Name: testapps.DefaultMySQLContainerName, Image: testapps.ApeCloudMySQLImage}).
- Create(&testCtx).GetObject()
- Expect(testapps.ChangeObjStatus(&testCtx, pod, func() {
- pod.Status.Conditions = []corev1.PodCondition{{
- Type: corev1.PodReady,
- Status: corev1.ConditionTrue,
- }}
- })).Should(Succeed())
- pods = append(pods, pod)
- }
- if intctrlutil.IsRSMEnabled() {
- rsm = testapps.NewRSMFactory(testCtx.DefaultNamespace, stsName, clusterName, compName).
- AddAnnotations(constant.KubeBlocksGenerationKey, strconv.FormatInt(cluster.Generation, 10)).
- SetReplicas(int32(3)).
- AddContainer(corev1.Container{Name: testapps.DefaultMySQLContainerName, Image: testapps.ApeCloudMySQLImage}).
- Create(&testCtx).GetObject()
- testk8s.InitRSMStatus(testCtx, rsm, controllerRevision)
- }
- Expect(testapps.ChangeObjStatus(&testCtx, cluster, func() {
- cluster.Status.ObservedGeneration = cluster.Generation
- })).Should(Succeed())
- })
-
- It("should set component status to failed if container is not ready and have error message", func() {
- Expect(mockContainerError(pods[0])).Should(Succeed())
-
- Expect(component.Status(*reqCtx, testCtx.Cli)).Should(Succeed())
- Expect(cluster.Status.Components[compName].Phase).Should(Equal(appsv1alpha1.FailedClusterCompPhase))
- })
-
- It("should set component status to running if container is ready", func() {
- Expect(testapps.ChangeObjStatus(&testCtx, statefulset, func() {
- testk8s.MockStatefulSetReady(statefulset)
- })).Should(Succeed())
- if intctrlutil.IsRSMEnabled() {
- Expect(testapps.ChangeObjStatus(&testCtx, rsm, func() {
- testk8s.MockRSMReady(rsm, pods...)
- })).Should(Succeed())
- }
-
- Expect(setPodRole(pods[0], "leader")).Should(Succeed())
- Expect(setPodRole(pods[1], "follower")).Should(Succeed())
- Expect(setPodRole(pods[2], "follower")).Should(Succeed())
-
- Expect(component.Status(*reqCtx, testCtx.Cli)).Should(Succeed())
- Expect(cluster.Status.Components[compName].Phase).Should(Equal(appsv1alpha1.RunningClusterCompPhase))
- })
- })
- })
-
- Context("with replicationset component", func() {
- BeforeEach(func() {
- clusterDef = testapps.NewClusterDefFactory(clusterDefName).
- AddComponentDef(testapps.ReplicationRedisComponent, compDefName).
- Create(&testCtx).GetObject()
-
- cluster = testapps.NewClusterFactory(testCtx.DefaultNamespace, clusterName, clusterDefName, clusterVersionName).
- AddComponent(compName, compDefName).
- SetReplicas(2).
- Create(&testCtx).GetObject()
-
- reqCtx = &intctrlutil.RequestCtx{
- Ctx: ctx,
- Log: log.FromContext(ctx).WithValues("cluster", clusterDef.Name),
- }
- dag = graph.NewDAG()
- component, err = NewComponent(*reqCtx, testCtx.Cli, clusterDef, nil, cluster, compName, dag)
- Expect(err).Should(Succeed())
- Expect(component).ShouldNot(BeNil())
- })
-
- It("should not change component if no deployment or pod exists", func() {
- Expect(component.Status(*reqCtx, testCtx.Cli)).Should(Succeed())
- Expect(cluster.Status.Components[compName].Phase).Should(BeEmpty())
- })
-
- Context("and with mocked statefulset & pod", func() {
- const (
- replicas = 2
- )
- var (
- statefulset *appsv1.StatefulSet
- pods []*corev1.Pod
- )
-
- BeforeEach(func() {
- stsName := clusterName + "-" + compName
- statefulset = testapps.NewStatefulSetFactory(testCtx.DefaultNamespace, stsName, clusterName, compName).
- AddAnnotations(constant.KubeBlocksGenerationKey, strconv.FormatInt(cluster.Generation, 10)).
- SetReplicas(int32(replicas)).
- AddContainer(corev1.Container{Name: testapps.DefaultRedisContainerName, Image: testapps.DefaultRedisImageName}).
- Create(&testCtx).GetObject()
- testk8s.InitStatefulSetStatus(testCtx, statefulset, controllerRevision)
- for i := 0; i < replicas; i++ {
- podName := fmt.Sprintf("%s-%d", stsName, i)
- podRole := "primary"
- if i > 0 {
- podRole = "secondary"
- }
- pod := testapps.NewPodFactory(testCtx.DefaultNamespace, podName).
- SetOwnerReferences("apps/v1", constant.StatefulSetKind, statefulset).
- AddAppInstanceLabel(clusterName).
- AddAppComponentLabel(compName).
- AddAppManangedByLabel().
- AddRoleLabel(podRole).
- AddControllerRevisionHashLabel(controllerRevision).
- AddContainer(corev1.Container{Name: testapps.DefaultRedisContainerName, Image: testapps.DefaultRedisImageName}).
- Create(&testCtx).GetObject()
- patch := client.MergeFrom(pod.DeepCopy())
- pod.Status.Conditions = []corev1.PodCondition{
- {
- Type: corev1.PodReady,
- Status: corev1.ConditionTrue,
- },
- }
- Expect(testCtx.Cli.Status().Patch(testCtx.Ctx, pod, patch)).Should(Succeed())
- pods = append(pods, pod)
- }
- if intctrlutil.IsRSMEnabled() {
- rsm = testapps.NewRSMFactory(testCtx.DefaultNamespace, stsName, clusterName, compName).
- AddAnnotations(constant.KubeBlocksGenerationKey, strconv.FormatInt(cluster.Generation, 10)).
- SetReplicas(int32(replicas)).
- AddContainer(corev1.Container{Name: testapps.DefaultRedisContainerName, Image: testapps.DefaultRedisImageName}).
- Create(&testCtx).GetObject()
- testk8s.InitRSMStatus(testCtx, rsm, controllerRevision)
- }
- Expect(testapps.ChangeObjStatus(&testCtx, cluster, func() {
- cluster.Status.ObservedGeneration = cluster.Generation
- })).Should(Succeed())
- })
-
- It("should set component status to failed if container is not ready and have error message", func() {
- Expect(mockContainerError(pods[0])).Should(Succeed())
-
- Expect(component.Status(*reqCtx, testCtx.Cli)).Should(Succeed())
- Expect(cluster.Status.Components[compName].Phase).Should(Equal(appsv1alpha1.FailedClusterCompPhase))
- })
-
- It("should set component status to running if container is ready", func() {
- Expect(testapps.ChangeObjStatus(&testCtx, statefulset, func() {
- testk8s.MockStatefulSetReady(statefulset)
- })).Should(Succeed())
- if intctrlutil.IsRSMEnabled() {
- Expect(testapps.ChangeObjStatus(&testCtx, rsm, func() {
- testk8s.MockRSMReady(rsm, pods...)
- })).Should(Succeed())
- }
-
- Expect(component.Status(*reqCtx, testCtx.Cli)).Should(Succeed())
- Expect(cluster.Status.Components[compName].Phase).Should(Equal(appsv1alpha1.RunningClusterCompPhase))
- })
- })
- })
-})
-
-func mockContainerError(pod *corev1.Pod) error {
- return testapps.ChangeObjStatus(&testCtx, pod, func() {
- pod.Status.ContainerStatuses = []corev1.ContainerStatus{
- {
- State: corev1.ContainerState{
- Waiting: &corev1.ContainerStateWaiting{
- Reason: "ImagePullBackOff",
- Message: "Back-off pulling image",
- },
- },
- },
- }
- pod.Status.Conditions = []corev1.PodCondition{
- {
- Type: corev1.ContainersReady,
- Status: corev1.ConditionFalse,
- LastTransitionTime: metav1.NewTime(time.Now().Add(-2 * time.Minute)),
- },
- }
- })
-}
-
-func setPodRole(pod *corev1.Pod, role string) error {
- return testapps.ChangeObj(&testCtx, pod, func(lpod *corev1.Pod) {
- lpod.Labels[constant.RoleLabelKey] = role
- })
-}
diff --git a/controllers/apps/components/types.go b/controllers/apps/components/types.go
index 98278bd3906..5c21d0629a3 100644
--- a/controllers/apps/components/types.go
+++ b/controllers/apps/components/types.go
@@ -20,62 +20,98 @@ along with this program. If not, see .
package components
import (
- "time"
+ "context"
+ "fmt"
"sigs.k8s.io/controller-runtime/pkg/client"
appsv1alpha1 "github.com/apecloud/kubeblocks/apis/apps/v1alpha1"
+ "github.com/apecloud/kubeblocks/internal/class"
+ "github.com/apecloud/kubeblocks/internal/constant"
+ types2 "github.com/apecloud/kubeblocks/internal/controller/client"
"github.com/apecloud/kubeblocks/internal/controller/component"
- ictrltypes "github.com/apecloud/kubeblocks/internal/controller/types"
+ "github.com/apecloud/kubeblocks/internal/controller/graph"
+ "github.com/apecloud/kubeblocks/internal/controller/plan"
intctrlutil "github.com/apecloud/kubeblocks/internal/controllerutil"
)
-const (
- // ComponentPhaseTransition the event reason indicates that the component transits to a new phase.
- ComponentPhaseTransition = "ComponentPhaseTransition"
-
- // PodContainerFailedTimeout the timeout for container of pod failures, the component phase will be set to Failed/Abnormal after this time.
- PodContainerFailedTimeout = 10 * time.Second
-
- // PodScheduledFailedTimeout timeout for scheduling failure.
- PodScheduledFailedTimeout = 30 * time.Second
-)
-
type Component interface {
GetName() string
GetNamespace() string
GetClusterName() string
- GetDefinitionName() string
- GetWorkloadType() appsv1alpha1.WorkloadType
GetCluster() *appsv1alpha1.Cluster
GetClusterVersion() *appsv1alpha1.ClusterVersion
GetSynthesizedComponent() *component.SynthesizedComponent
- GetConsensusSpec() *appsv1alpha1.ConsensusSetSpec
-
- GetMatchingLabels() client.MatchingLabels
-
- GetPhase() appsv1alpha1.ClusterComponentPhase
- // GetStatus() appsv1alpha1.ClusterComponentStatus
-
- // GetBuiltObjects returns all objects that will be created by this component
- GetBuiltObjects(reqCtx intctrlutil.RequestCtx, cli client.Client) ([]client.Object, error)
-
Create(reqCtx intctrlutil.RequestCtx, cli client.Client) error
Delete(reqCtx intctrlutil.RequestCtx, cli client.Client) error
Update(reqCtx intctrlutil.RequestCtx, cli client.Client) error
Status(reqCtx intctrlutil.RequestCtx, cli client.Client) error
+}
- Restart(reqCtx intctrlutil.RequestCtx, cli client.Client) error
-
- ExpandVolume(reqCtx intctrlutil.RequestCtx, cli client.Client) error
-
- HorizontalScale(reqCtx intctrlutil.RequestCtx, cli client.Client) error
-
- // TODO(impl): impl-related, replace them with component workload
- SetWorkload(obj client.Object, action *ictrltypes.LifecycleAction, parent *ictrltypes.LifecycleVertex)
- AddResource(obj client.Object, action *ictrltypes.LifecycleAction, parent *ictrltypes.LifecycleVertex) *ictrltypes.LifecycleVertex
+func NewComponent(reqCtx intctrlutil.RequestCtx,
+ cli client.Client,
+ definition *appsv1alpha1.ClusterDefinition,
+ version *appsv1alpha1.ClusterVersion,
+ cluster *appsv1alpha1.Cluster,
+ compName string,
+ dag *graph.DAG) (Component, error) {
+ var compDef *appsv1alpha1.ClusterComponentDefinition
+ var compVer *appsv1alpha1.ClusterComponentVersion
+ compSpec := cluster.Spec.GetComponentByName(compName)
+ if compSpec != nil {
+ compDef = definition.GetComponentDefByName(compSpec.ComponentDefRef)
+ if compDef == nil {
+ return nil, fmt.Errorf("referenced component definition does not exist, cluster: %s, component: %s, component definition ref:%s",
+ cluster.Name, compSpec.Name, compSpec.ComponentDefRef)
+ }
+ if version != nil {
+ compVer = version.Spec.GetDefNameMappingComponents()[compSpec.ComponentDefRef]
+ }
+ } else {
+ compDef = definition.GetComponentDefByName(compName)
+ if version != nil {
+ compVer = version.Spec.GetDefNameMappingComponents()[compName]
+ }
+ }
+
+ if compDef == nil {
+ return nil, nil
+ }
+
+ clsMgr, err := getClassManager(reqCtx.Ctx, cli, cluster)
+ if err != nil {
+ return nil, err
+ }
+ serviceReferences, err := plan.GenServiceReferences(reqCtx, cli, cluster, compDef, compSpec)
+ if err != nil {
+ return nil, err
+ }
+
+ synthesizedComp, err := component.BuildComponent(reqCtx, clsMgr, cluster, definition, compDef, compSpec, serviceReferences, compVer)
+ if err != nil {
+ return nil, err
+ }
+ if synthesizedComp == nil {
+ return nil, nil
+ }
+
+ return newRSMComponent(cli, reqCtx.Recorder, cluster, version, synthesizedComp, dag), nil
}
-type ComponentWorkload interface{}
+func getClassManager(ctx context.Context, cli types2.ReadonlyClient, cluster *appsv1alpha1.Cluster) (*class.Manager, error) {
+ var classDefinitionList appsv1alpha1.ComponentClassDefinitionList
+ ml := []client.ListOption{
+ client.MatchingLabels{constant.ClusterDefLabelKey: cluster.Spec.ClusterDefRef},
+ }
+ if err := cli.List(ctx, &classDefinitionList, ml...); err != nil {
+ return nil, err
+ }
+
+ var constraintList appsv1alpha1.ComponentResourceConstraintList
+ if err := cli.List(ctx, &constraintList); err != nil {
+ return nil, err
+ }
+ return class.NewManager(classDefinitionList, constraintList)
+}
diff --git a/controllers/apps/components/utils.go b/controllers/apps/components/utils.go
index d33462343d2..e9a025440a2 100644
--- a/controllers/apps/components/utils.go
+++ b/controllers/apps/components/utils.go
@@ -24,7 +24,6 @@ import (
"errors"
"fmt"
"reflect"
- "sort"
"strconv"
"strings"
"time"
@@ -55,7 +54,7 @@ var (
)
func listObjWithLabelsInNamespace[T generics.Object, PT generics.PObject[T], L generics.ObjList[T], PL generics.PObjList[T, L]](
- ctx context.Context, cli client.Client, _ func(T, L), namespace string, labels client.MatchingLabels) ([]PT, error) {
+ ctx context.Context, cli client.Client, _ func(T, PT, L, PL), namespace string, labels client.MatchingLabels) ([]PT, error) {
var objList L
if err := cli.List(ctx, PL(&objList), labels, client.InNamespace(namespace)); err != nil {
return nil, err
@@ -73,14 +72,6 @@ func listRSMOwnedByComponent(ctx context.Context, cli client.Client, namespace s
return listObjWithLabelsInNamespace(ctx, cli, generics.RSMSignature, namespace, labels)
}
-func listStsOwnedByComponent(ctx context.Context, cli client.Client, namespace string, labels client.MatchingLabels) ([]*appsv1.StatefulSet, error) {
- return listObjWithLabelsInNamespace(ctx, cli, generics.StatefulSetSignature, namespace, labels)
-}
-
-func listDeployOwnedByComponent(ctx context.Context, cli client.Client, namespace string, labels client.MatchingLabels) ([]*appsv1.Deployment, error) {
- return listObjWithLabelsInNamespace(ctx, cli, generics.DeploymentSignature, namespace, labels)
-}
-
func listPodOwnedByComponent(ctx context.Context, cli client.Client, namespace string, labels client.MatchingLabels) ([]*corev1.Pod, error) {
return listObjWithLabelsInNamespace(ctx, cli, generics.PodSignature, namespace, labels)
}
@@ -353,75 +344,6 @@ func getCompRelatedObjectList(ctx context.Context,
return podList, nil
}
-// availableReplicasAreConsistent checks if expected replicas number of component is consistent with
-// the number of available workload replicas.
-func availableReplicasAreConsistent(componentReplicas, podCount, workloadAvailableReplicas int32) bool {
- return workloadAvailableReplicas == componentReplicas && componentReplicas == podCount
-}
-
-// getPhaseWithNoAvailableReplicas gets the component phase when the workload of component has no available replicas.
-func getPhaseWithNoAvailableReplicas(componentReplicas int32) appsv1alpha1.ClusterComponentPhase {
- if componentReplicas == 0 {
- return ""
- }
- return appsv1alpha1.FailedClusterCompPhase
-}
-
-// getComponentPhaseWhenPodsNotReady gets the component phase when pods of component are not ready.
-func getComponentPhaseWhenPodsNotReady(podList *corev1.PodList,
- workload metav1.Object,
- componentReplicas,
- availableReplicas int32,
- checkLeaderIsReady func(pod *corev1.Pod, workload metav1.Object) bool,
- checkFailedPodRevision func(pod *corev1.Pod, workload metav1.Object) bool) appsv1alpha1.ClusterComponentPhase {
- podCount := len(podList.Items)
- if podCount == 0 || availableReplicas == 0 {
- return getPhaseWithNoAvailableReplicas(componentReplicas)
- }
- var (
- existLatestRevisionFailedPod bool
- leaderIsReady bool
- )
- for _, v := range podList.Items {
- // if the pod is terminating, ignore it
- if v.DeletionTimestamp != nil {
- return ""
- }
- if checkLeaderIsReady == nil || checkLeaderIsReady(&v, workload) {
- leaderIsReady = true
- }
- if checkFailedPodRevision != nil && checkFailedPodRevision(&v, workload) {
- existLatestRevisionFailedPod = true
- }
- }
- return getCompPhaseByConditions(existLatestRevisionFailedPod, leaderIsReady,
- componentReplicas, int32(podCount), availableReplicas)
-}
-
-// getCompPhaseByConditions gets the component phase according to the following conditions:
-// 1. if the failed pod is not controlled by the latest revision, ignore it.
-// 2. if the primary replicas are not available, the component is failed.
-// 3. finally if expected replicas number of component is inconsistent with
-// the number of available workload replicas, the component is abnormal.
-func getCompPhaseByConditions(existLatestRevisionFailedPod bool,
- primaryReplicasAvailable bool,
- compReplicas,
- podCount,
- availableReplicas int32) appsv1alpha1.ClusterComponentPhase {
- // if the failed pod is not controlled by the latest revision, ignore it.
- if !existLatestRevisionFailedPod {
- return ""
- }
- if !primaryReplicasAvailable {
- return appsv1alpha1.FailedClusterCompPhase
- }
- // checks if expected replicas number of component is consistent with the number of available workload replicas.
- if !availableReplicasAreConsistent(compReplicas, podCount, availableReplicas) {
- return appsv1alpha1.AbnormalClusterCompPhase
- }
- return ""
-}
-
// parseCustomLabelPattern parses the custom label pattern to GroupVersionKind.
func parseCustomLabelPattern(pattern string) (schema.GroupVersionKind, error) {
patterns := strings.Split(pattern, "/")
@@ -442,37 +364,12 @@ func parseCustomLabelPattern(pattern string) (schema.GroupVersionKind, error) {
return schema.GroupVersionKind{}, fmt.Errorf("invalid pattern %s", pattern)
}
-// SortPods sorts pods by their role priority
-func SortPods(pods []corev1.Pod, priorityMap map[string]int, idLabelKey string) {
- // make a Serial pod list,
- // e.g.: unknown -> empty -> learner -> follower1 -> follower2 -> leader, with follower1.Name < follower2.Name
- sort.SliceStable(pods, func(i, j int) bool {
- roleI := pods[i].Labels[idLabelKey]
- roleJ := pods[j].Labels[idLabelKey]
- if priorityMap[roleI] == priorityMap[roleJ] {
- _, ordinal1 := intctrlutil.GetParentNameAndOrdinal(&pods[i])
- _, ordinal2 := intctrlutil.GetParentNameAndOrdinal(&pods[j])
- return ordinal1 < ordinal2
- }
- return priorityMap[roleI] < priorityMap[roleJ]
- })
-}
-
// replaceKBEnvPlaceholderTokens replaces the placeholder tokens in the string strToReplace with builtInEnvMap and return new string.
func replaceKBEnvPlaceholderTokens(clusterName, uid, componentName, strToReplace string) string {
builtInEnvMap := componentutil.GetReplacementMapForBuiltInEnv(clusterName, uid, componentName)
return componentutil.ReplaceNamedVars(builtInEnvMap, strToReplace, -1, true)
}
-// getRunningPods gets the running pods of the specified statefulSet.
-func getRunningPods(ctx context.Context, cli client.Client, obj client.Object) ([]corev1.Pod, error) {
- sts := convertToStatefulSet(obj)
- if sts == nil || sts.Generation != sts.Status.ObservedGeneration {
- return nil, nil
- }
- return GetPodListByStatefulSet(ctx, cli, sts)
-}
-
// resolvePodSpecDefaultFields set default value for some known fields of proto PodSpec @pobj.
func resolvePodSpecDefaultFields(obj corev1.PodSpec, pobj *corev1.PodSpec) {
resolveVolume := func(v corev1.Volume, vv *corev1.Volume) {
diff --git a/controllers/apps/components/utils_test.go b/controllers/apps/components/utils_test.go
index 4571e6f612d..fd38f183b96 100644
--- a/controllers/apps/components/utils_test.go
+++ b/controllers/apps/components/utils_test.go
@@ -40,10 +40,8 @@ import (
"github.com/apecloud/kubeblocks/internal/controller/component"
"github.com/apecloud/kubeblocks/internal/controller/graph"
"github.com/apecloud/kubeblocks/internal/controller/types"
- intctrlutil "github.com/apecloud/kubeblocks/internal/controllerutil"
"github.com/apecloud/kubeblocks/internal/generics"
testapps "github.com/apecloud/kubeblocks/internal/testutil/apps"
- testk8s "github.com/apecloud/kubeblocks/internal/testutil/k8s"
)
func TestIsFailedOrAbnormal(t *testing.T) {
@@ -65,82 +63,7 @@ func TestIsProbeTimeout(t *testing.T) {
}
}
-func TestGetComponentPhase(t *testing.T) {
- var (
- isFailed = true
- isAbnormal = true
- )
- getComponentPhase := func(isFailed, isAbnormal bool) appsv1alpha1.ClusterComponentPhase {
- var componentPhase appsv1alpha1.ClusterComponentPhase
- if isFailed {
- componentPhase = appsv1alpha1.FailedClusterCompPhase
- } else if isAbnormal {
- componentPhase = appsv1alpha1.AbnormalClusterCompPhase
- }
- return componentPhase
- }
- status := getComponentPhase(isFailed, isAbnormal)
- if status != appsv1alpha1.FailedClusterCompPhase {
- t.Error("function getComponentPhase should return Failed")
- }
- isFailed = false
- status = getComponentPhase(isFailed, isAbnormal)
- if status != appsv1alpha1.AbnormalClusterCompPhase {
- t.Error("function getComponentPhase should return Abnormal")
- }
- isAbnormal = false
- status = getComponentPhase(isFailed, isAbnormal)
- if status != "" {
- t.Error(`function getComponentPhase should return ""`)
- }
-}
-
-func TestGetPhaseWithNoAvailableReplicas(t *testing.T) {
- status := getPhaseWithNoAvailableReplicas(int32(0))
- if status != "" {
- t.Error(`function getComponentPhase should return ""`)
- }
- status = getPhaseWithNoAvailableReplicas(int32(2))
- if status != appsv1alpha1.FailedClusterCompPhase {
- t.Error(`function getComponentPhase should return "Failed"`)
- }
-}
-
-func TestAvailableReplicasAreConsistent(t *testing.T) {
- isConsistent := availableReplicasAreConsistent(int32(1), int32(1), int32(1))
- if !isConsistent {
- t.Error(`function getComponentPhase should return "true"`)
- }
- isConsistent = availableReplicasAreConsistent(int32(1), int32(2), int32(1))
- if isConsistent {
- t.Error(`function getComponentPhase should return "false"`)
- }
-}
-
-func TestGetCompPhaseByConditions(t *testing.T) {
- existLatestRevisionFailedPod := true
- primaryReplicaIsReady := true
- phase := getCompPhaseByConditions(existLatestRevisionFailedPod, primaryReplicaIsReady, int32(1), int32(1), int32(1))
- if phase != "" {
- t.Error(`function getComponentPhase should return ""`)
- }
- phase = getCompPhaseByConditions(existLatestRevisionFailedPod, primaryReplicaIsReady, int32(2), int32(1), int32(1))
- if phase != appsv1alpha1.AbnormalClusterCompPhase {
- t.Error(`function getComponentPhase should return "Abnormal"`)
- }
- primaryReplicaIsReady = false
- phase = getCompPhaseByConditions(existLatestRevisionFailedPod, primaryReplicaIsReady, int32(2), int32(1), int32(1))
- if phase != appsv1alpha1.FailedClusterCompPhase {
- t.Error(`function getComponentPhase should return "Failed"`)
- }
- existLatestRevisionFailedPod = false
- phase = getCompPhaseByConditions(existLatestRevisionFailedPod, primaryReplicaIsReady, int32(2), int32(1), int32(1))
- if phase != "" {
- t.Error(`function getComponentPhase should return ""`)
- }
-}
-
-var _ = Describe("Consensus Component", func() {
+var _ = Describe("Component", func() {
var (
randomStr = testCtx.GetRandomStr()
clusterDefName = "mysql-clusterdef-" + randomStr
@@ -175,8 +98,8 @@ var _ = Describe("Consensus Component", func() {
AfterEach(cleanAll)
- Context("Consensus Component test", func() {
- It("Consensus Component test", func() {
+ Context("Component test", func() {
+ It("Component test", func() {
By(" init cluster, statefulSet, pods")
_, _, cluster := testapps.InitClusterWithHybridComps(&testCtx, clusterDefName,
clusterVersionName, clusterName, statelessCompName, "stateful", consensusCompName)
@@ -246,38 +169,6 @@ var _ = Describe("Consensus Component", func() {
delete(podNoLabel.Labels, constant.KBAppComponentLabelKey)
_, _, err = GetComponentInfoByPod(ctx, k8sClient, *cluster, podNoLabel)
Expect(err).ShouldNot(Succeed())
-
- By("test getComponentPhaseWhenPodsNotReady function")
- consensusComp := cluster.Spec.GetComponentByName(consensusCompName)
- checkExistFailedPodOfLatestRevision := func(pod *corev1.Pod, workload metav1.Object) bool {
- sts := workload.(*appsv1.StatefulSet)
- return !intctrlutil.PodIsReady(pod) && intctrlutil.PodIsControlledByLatestRevision(pod, sts)
- }
- // component phase should be Failed when available replicas is 0
- phase := getComponentPhaseWhenPodsNotReady(podList, sts, consensusComp.Replicas,
- sts.Status.AvailableReplicas, nil, checkExistFailedPodOfLatestRevision)
- Expect(phase).Should(Equal(appsv1alpha1.FailedClusterCompPhase))
-
- // mock available replicas to component replicas
- Expect(testapps.ChangeObjStatus(&testCtx, sts, func() {
- testk8s.MockStatefulSetReady(sts)
- })).Should(Succeed())
- phase = getComponentPhaseWhenPodsNotReady(podList, sts, consensusComp.Replicas,
- sts.Status.AvailableReplicas, nil, checkExistFailedPodOfLatestRevision)
- Expect(len(phase) == 0).Should(BeTrue())
-
- // mock component is abnormal
- pod := &podList.Items[0]
- Expect(testapps.ChangeObjStatus(&testCtx, pod, func() {
- pod.Status.Conditions = nil
- })).Should(Succeed())
- Expect(testapps.ChangeObjStatus(&testCtx, sts, func() {
- sts.Status.AvailableReplicas = *sts.Spec.Replicas - 1
- })).Should(Succeed())
- phase = getComponentPhaseWhenPodsNotReady(podList, sts, consensusComp.Replicas,
- sts.Status.AvailableReplicas, nil, checkExistFailedPodOfLatestRevision)
- Expect(phase).Should(Equal(appsv1alpha1.AbnormalClusterCompPhase))
-
})
It("test GetComponentInfoByPod with no cluster componentSpec", func() {
@@ -402,7 +293,7 @@ var _ = Describe("Component utils test", func() {
SetOwnerReferences("apps/v1", constant.StatefulSetKind, nil).
AddAppInstanceLabel(clusterName).
AddAppComponentLabel(compName).
- AddAppManangedByLabel().
+ AddAppManagedByLabel().
AddRoleLabel(role).
AddConsensusSetAccessModeLabel(mode).
AddControllerRevisionHashLabel("").
diff --git a/controllers/apps/configuration/config_related_helper.go b/controllers/apps/configuration/config_related_helper.go
index 932f1a11317..1686289c012 100644
--- a/controllers/apps/configuration/config_related_helper.go
+++ b/controllers/apps/configuration/config_related_helper.go
@@ -35,7 +35,7 @@ import (
"github.com/apecloud/kubeblocks/internal/generics"
)
-func retrieveRelatedComponentsByConfigmap[T generics.Object, L generics.ObjList[T], PL generics.PObjList[T, L]](cli client.Client, ctx context.Context, configSpecName string, _ func(T, L), cfg client.ObjectKey, opts ...client.ListOption) ([]T, []string, error) {
+func retrieveRelatedComponentsByConfigmap[T generics.Object, PT generics.PObject[T], L generics.ObjList[T], PL generics.PObjList[T, L]](cli client.Client, ctx context.Context, configSpecName string, _ func(T, PT, L, PL), cfg client.ObjectKey, opts ...client.ListOption) ([]T, []string, error) {
var objList L
if err := cli.List(ctx, PL(&objList), opts...); err != nil {
return nil, nil, err
diff --git a/controllers/apps/configuration/configuration_controller.go b/controllers/apps/configuration/configuration_controller.go
index 28d4a1d56ec..9208d02a498 100644
--- a/controllers/apps/configuration/configuration_controller.go
+++ b/controllers/apps/configuration/configuration_controller.go
@@ -21,6 +21,7 @@ package configuration
import (
"context"
+ "fmt"
"strconv"
"time"
@@ -46,7 +47,7 @@ type ConfigurationReconciler struct {
Recorder record.EventRecorder
}
-const reconcileInterval = time.Millisecond * 10
+const reconcileInterval = time.Second * 2
//+kubebuilder:rbac:groups=apps.kubeblocks.io,resources=configurations,verbs=get;list;watch;create;update;patch;delete
//+kubebuilder:rbac:groups=apps.kubeblocks.io,resources=configurations/status,verbs=get;update;patch
@@ -102,7 +103,11 @@ func (r *ConfigurationReconciler) Reconcile(ctx context.Context, req ctrl.Reques
return intctrlutil.CheckedRequeueWithError(err, reqCtx.Log, "failed to get related object.")
}
- if err := r.runTasks(reqCtx, configuration, fetcherTask, tasks); err != nil {
+ if fetcherTask.ClusterComObj == nil || fetcherTask.ClusterDefComObj == nil {
+ return r.failWithInvalidComponent(configuration, reqCtx)
+ }
+
+ if err := r.runTasks(TaskContext{configuration, reqCtx, fetcherTask}, tasks); err != nil {
return intctrlutil.CheckedRequeueWithError(err, reqCtx.Log, "failed to run configuration reconcile task.")
}
if !isAllReady(configuration) {
@@ -111,59 +116,70 @@ func (r *ConfigurationReconciler) Reconcile(ctx context.Context, req ctrl.Reques
return intctrlutil.Reconciled()
}
+func (r *ConfigurationReconciler) failWithInvalidComponent(configuration *appsv1alpha1.Configuration, reqCtx intctrlutil.RequestCtx) (ctrl.Result, error) {
+ msg := fmt.Sprintf("not found cluster component or cluster definition component: [%s]", configuration.Spec.ComponentName)
+ reqCtx.Log.Error(fmt.Errorf(msg), "")
+ patch := client.MergeFrom(configuration.DeepCopy())
+ configuration.Status.Message = msg
+ if err := r.Client.Status().Patch(reqCtx.Ctx, configuration, patch); err != nil {
+ return intctrlutil.CheckedRequeueWithError(err, reqCtx.Log, "failed to update configuration status.")
+ }
+ return intctrlutil.Reconciled()
+}
+
func isAllReady(configuration *appsv1alpha1.Configuration) bool {
for _, item := range configuration.Spec.ConfigItemDetails {
itemStatus := configuration.Status.GetItemStatus(item.Name)
- if itemStatus == nil || itemStatus.Phase != appsv1alpha1.CFinishedPhase {
+ if itemStatus != nil && !isFinishStatus(itemStatus.Phase) {
return false
}
}
return true
}
-func (r *ConfigurationReconciler) runTasks(
- reqCtx intctrlutil.RequestCtx,
- configuration *appsv1alpha1.Configuration,
- fetcher *Task,
- tasks []Task) (err error) {
- var errs []error
- var synthesizedComp *component.SynthesizedComponent
-
- synthesizedComp, err = component.BuildComponent(reqCtx, nil,
- fetcher.ClusterObj,
- fetcher.ClusterDefObj,
- fetcher.ClusterDefComObj,
- fetcher.ClusterComObj,
+func (r *ConfigurationReconciler) runTasks(taskCtx TaskContext, tasks []Task) (err error) {
+ var (
+ errs []error
+ synthesizedComp *component.SynthesizedComponent
+
+ ctx = taskCtx.reqCtx.Ctx
+ configuration = taskCtx.configuration
+ )
+
+ synthesizedComp, err = component.BuildComponent(taskCtx.reqCtx,
nil,
- fetcher.ClusterVerComObj)
+ taskCtx.fetcher.ClusterObj,
+ taskCtx.fetcher.ClusterDefObj,
+ taskCtx.fetcher.ClusterDefComObj,
+ taskCtx.fetcher.ClusterComObj,
+ nil,
+ taskCtx.fetcher.ClusterVerComObj)
if err != nil {
- return
+ return err
}
- revision := strconv.FormatInt(configuration.GetGeneration(), 10)
+ // TODO manager multiple version
patch := client.MergeFrom(configuration.DeepCopy())
+ revision := strconv.FormatInt(configuration.GetGeneration(), 10)
for _, task := range tasks {
- if err := task.Do(fetcher, synthesizedComp, revision); err != nil {
+ task.Status.UpdateRevision = revision
+ if err := task.Do(taskCtx.fetcher, synthesizedComp, revision); err != nil {
task.Status.Phase = appsv1alpha1.CMergeFailedPhase
task.Status.Message = cfgutil.ToPointer(err.Error())
errs = append(errs, err)
continue
}
- if err := task.SyncStatus(fetcher, task.Status); err != nil {
- task.Status.Phase = appsv1alpha1.CFailedPhase
- task.Status.Message = cfgutil.ToPointer(err.Error())
- errs = append(errs, err)
- }
}
+ configuration.Status.Message = ""
if len(errs) > 0 {
configuration.Status.Message = utilerrors.NewAggregate(errs).Error()
}
- if err := r.Client.Status().Patch(reqCtx.Ctx, configuration, patch); err != nil {
+ if err := r.Client.Status().Patch(ctx, configuration, patch); err != nil {
errs = append(errs, err)
}
if len(errs) == 0 {
- return
+ return nil
}
return utilerrors.NewAggregate(errs)
}
@@ -177,6 +193,10 @@ func (r *ConfigurationReconciler) SetupWithManager(mgr ctrl.Manager) error {
}
func fromItemStatus(ctx intctrlutil.RequestCtx, status *appsv1alpha1.ConfigurationStatus, item appsv1alpha1.ConfigurationItemDetail) *appsv1alpha1.ConfigurationItemDetailStatus {
+ if item.ConfigSpec == nil {
+ ctx.Log.WithName(item.Name).Error(core.MakeError("configSpec phase is not ready and pass: %v", item), "")
+ return nil
+ }
for i := range status.ConfigurationItemStatus {
itemStatus := &status.ConfigurationItemStatus[i]
switch {
@@ -193,11 +213,11 @@ func fromItemStatus(ctx intctrlutil.RequestCtx, status *appsv1alpha1.Configurati
}
func isReconcileStatus(phase appsv1alpha1.ConfigurationPhase) bool {
- return phase == appsv1alpha1.CRunningPhase ||
- phase == appsv1alpha1.CInitPhase ||
- phase == appsv1alpha1.CPendingPhase ||
- phase == appsv1alpha1.CMergedPhase ||
- phase == appsv1alpha1.CMergeFailedPhase ||
- phase == appsv1alpha1.CUpgradingPhase ||
- phase == appsv1alpha1.CFinishedPhase
+ return phase != "" &&
+ phase != appsv1alpha1.CCreatingPhase &&
+ phase != appsv1alpha1.CDeletingPhase
+}
+
+func isFinishStatus(phase appsv1alpha1.ConfigurationPhase) bool {
+ return phase == appsv1alpha1.CFinishedPhase || phase == appsv1alpha1.CFailedAndPausePhase
}
diff --git a/controllers/apps/configuration/configuration_controller_test.go b/controllers/apps/configuration/configuration_controller_test.go
index 57a21717b44..8e711818f1e 100644
--- a/controllers/apps/configuration/configuration_controller_test.go
+++ b/controllers/apps/configuration/configuration_controller_test.go
@@ -86,9 +86,33 @@ var _ = Describe("Configuration Controller", func() {
g.Expect(k8sClient.Get(ctx, cfgKey, cfg)).Should(Succeed())
itemStatus := cfg.Status.GetItemStatus(configSpecName)
g.Expect(itemStatus).ShouldNot(BeNil())
+ g.Expect(itemStatus.UpdateRevision).Should(BeEquivalentTo("2"))
g.Expect(itemStatus.Phase).Should(BeEquivalentTo(appsv1alpha1.CFinishedPhase))
}, time.Second*60, time.Second*1).Should(Succeed())
})
+
+ It("Invalid component test", func() {
+ _, _, clusterObj, clusterVersionObj, synthesizedComp := mockReconcileResource()
+
+ cfgKey := client.ObjectKey{
+ Name: core.GenerateComponentConfigurationName(clusterName, "invalid-component"),
+ Namespace: testCtx.DefaultNamespace,
+ }
+
+ Expect(initConfiguration(&intctrlutil.ResourceCtx{
+ Client: k8sClient,
+ Context: ctx,
+ Namespace: testCtx.DefaultNamespace,
+ ClusterName: clusterName,
+ ComponentName: "invalid-component",
+ }, synthesizedComp, clusterObj, clusterVersionObj)).Should(Succeed())
+
+ Eventually(func(g Gomega) {
+ cfg := &appsv1alpha1.Configuration{}
+ g.Expect(k8sClient.Get(ctx, cfgKey, cfg)).Should(Succeed())
+ g.Expect(cfg.Status.Message).Should(ContainSubstring("not found cluster component"))
+ }, time.Second*60, time.Second*1).Should(Succeed())
+ })
})
})
diff --git a/controllers/apps/configuration/configuration_test.go b/controllers/apps/configuration/configuration_test.go
index e85b507d0e1..8bb5bbe0548 100644
--- a/controllers/apps/configuration/configuration_test.go
+++ b/controllers/apps/configuration/configuration_test.go
@@ -81,9 +81,15 @@ func mockConfigResource() (*corev1.ConfigMap, *appsv1alpha1.ConfigConstraint) {
configuration := builder.NewConfigurationBuilder(testCtx.DefaultNamespace, core.GenerateComponentConfigurationName(clusterName, statefulCompName)).
ClusterRef(clusterName).
Component(statefulCompName).
- ClusterDefRef(clusterDefName).
- ClusterVerRef(clusterVersionName).
- AddConfigurationItem(configSpecName).
+ AddConfigurationItem(appsv1alpha1.ComponentConfigSpec{
+ ComponentTemplateSpec: appsv1alpha1.ComponentTemplateSpec{
+ Name: configSpecName,
+ TemplateRef: configmap.Name,
+ Namespace: configmap.Namespace,
+ VolumeName: configVolumeName,
+ },
+ ConfigConstraintRef: constraint.Name,
+ }).
GetObject()
Expect(testCtx.CreateObj(testCtx.Ctx, configuration)).Should(Succeed())
@@ -113,13 +119,11 @@ func mockReconcileResource() (*corev1.ConfigMap, *appsv1alpha1.ConfigConstraint,
clusterDefObj.Name, clusterVersionObj.Name).
AddComponent(statefulCompName, statefulCompDefName).Create(&testCtx).GetObject()
- container := corev1.Container{
- Name: "mock-container",
- VolumeMounts: []corev1.VolumeMount{{
+ container := *builder.NewContainerBuilder("mock-container").
+ AddVolumeMounts(corev1.VolumeMount{
Name: configVolumeName,
MountPath: "/mnt/config",
- }},
- }
+ }).GetObject()
_ = testapps.NewStatefulSetFactory(testCtx.DefaultNamespace, statefulSetName, clusterObj.Name, statefulCompName).
AddConfigmapVolume(configVolumeName, configmap.Name).
AddContainer(container).
diff --git a/controllers/apps/configuration/parallel_upgrade_policy_test.go b/controllers/apps/configuration/parallel_upgrade_policy_test.go
index 27e6c12a905..e1d89971bb0 100644
--- a/controllers/apps/configuration/parallel_upgrade_policy_test.go
+++ b/controllers/apps/configuration/parallel_upgrade_policy_test.go
@@ -66,6 +66,7 @@ var _ = Describe("Reconfigure ParallelPolicy", func() {
return reconfigureClient, nil
}),
withMockStatefulSet(3, nil),
+ withClusterComponent(3),
withConfigSpec("for_test", map[string]string{
"a": "b",
}),
@@ -94,6 +95,7 @@ var _ = Describe("Reconfigure ParallelPolicy", func() {
return reconfigureClient, nil
}),
withMockStatefulSet(3, nil),
+ withClusterComponent(3),
withConfigSpec("for_test", map[string]string{
"a": "b",
}),
@@ -133,6 +135,7 @@ var _ = Describe("Reconfigure ParallelPolicy", func() {
return reconfigureClient, nil
}),
withMockStatefulSet(3, nil),
+ withClusterComponent(3),
withConfigSpec("for_test", map[string]string{
"a": "b",
}),
@@ -174,6 +177,7 @@ var _ = Describe("Reconfigure ParallelPolicy", func() {
return reconfigureClient, nil
}),
withMockStatefulSet(3, nil),
+ withClusterComponent(3),
withConfigSpec("for_test", map[string]string{
"a": "b",
}),
@@ -203,6 +207,7 @@ var _ = Describe("Reconfigure ParallelPolicy", func() {
withConfigSpec("for_test", map[string]string{
"key": "value",
}),
+ withClusterComponent(2),
withCDComponent(appsv1alpha1.Stateless, []appsv1alpha1.ComponentConfigSpec{{
ComponentTemplateSpec: appsv1alpha1.ComponentTemplateSpec{
Name: "for_test",
diff --git a/controllers/apps/configuration/policy_util.go b/controllers/apps/configuration/policy_util.go
index 4f054c325bb..98fab144e7a 100644
--- a/controllers/apps/configuration/policy_util.go
+++ b/controllers/apps/configuration/policy_util.go
@@ -31,9 +31,11 @@ import (
"sigs.k8s.io/controller-runtime/pkg/client"
"github.com/apecloud/kubeblocks/controllers/apps/components"
+ "github.com/apecloud/kubeblocks/internal/common"
"github.com/apecloud/kubeblocks/internal/configuration/core"
cfgproto "github.com/apecloud/kubeblocks/internal/configuration/proto"
"github.com/apecloud/kubeblocks/internal/constant"
+ "github.com/apecloud/kubeblocks/internal/controller/rsm"
intctrlutil "github.com/apecloud/kubeblocks/internal/controllerutil"
viper "github.com/apecloud/kubeblocks/internal/viperx"
)
@@ -57,7 +59,13 @@ func getReplicationSetPods(params reconfigureParams) ([]corev1.Pod, error) {
func GetComponentPods(params reconfigureParams) ([]corev1.Pod, error) {
componentPods := make([]corev1.Pod, 0)
for i := range params.ComponentUnits {
- pods, err := components.GetPodListByStatefulSet(params.Ctx.Ctx, params.Client, ¶ms.ComponentUnits[i])
+ pods, err := common.GetPodListByStatefulSetWithSelector(params.Ctx.Ctx,
+ params.Client,
+ ¶ms.ComponentUnits[i],
+ client.MatchingLabels{
+ constant.KBAppComponentLabelKey: params.ClusterComponent.Name,
+ constant.AppInstanceLabelKey: params.Cluster.Name,
+ })
if err != nil {
return nil, err
}
@@ -87,8 +95,7 @@ func getStatefulSetPods(params reconfigureParams) ([]corev1.Pod, error) {
return nil, core.MakeError("statefulSet component require only one statefulset, actual %d components", len(params.ComponentUnits))
}
- stsObj := ¶ms.ComponentUnits[0]
- pods, err := components.GetPodListByStatefulSet(params.Ctx.Ctx, params.Client, stsObj)
+ pods, err := GetComponentPods(params)
if err != nil {
return nil, err
}
@@ -110,19 +117,21 @@ func getConsensusPods(params reconfigureParams) ([]corev1.Pod, error) {
return nil, nil
}
- stsObj := ¶ms.ComponentUnits[0]
- pods, err := components.GetPodListByStatefulSet(params.Ctx.Ctx, params.Client, stsObj)
+ pods, err := GetComponentPods(params)
+ // stsObj := ¶ms.ComponentUnits[0]
+ // pods, err := components.GetPodListByStatefulSetWithSelector(params.Ctx.Ctx, params.Client, stsObj, client.MatchingLabels{
+ // constant.KBAppComponentLabelKey: params.ClusterComponent.Name,
+ // constant.AppInstanceLabelKey: params.Cluster.Name,
+ // })
if err != nil {
return nil, err
}
// TODO: should resolve the dependency on consensus module
- components.SortPods(pods, components.ComposeRolePriorityMap(params.Component.ConsensusSpec), constant.RoleLabelKey)
- r := make([]corev1.Pod, 0, len(pods))
- for i := len(pods); i > 0; i-- {
- r = append(r, pods[i-1:i]...)
+ if params.Component.RSMSpec != nil {
+ rsm.SortPods(pods, rsm.ComposeRolePriorityMap(params.Component.RSMSpec.Roles), true)
}
- return r, nil
+ return pods, nil
}
// TODO commonOnlineUpdateWithPod migrate to sql command pipeline
diff --git a/controllers/apps/configuration/policy_util_test.go b/controllers/apps/configuration/policy_util_test.go
index cc07a086c4a..e7635f9e530 100644
--- a/controllers/apps/configuration/policy_util_test.go
+++ b/controllers/apps/configuration/policy_util_test.go
@@ -35,6 +35,7 @@ import (
"sigs.k8s.io/controller-runtime/pkg/log"
appsv1alpha1 "github.com/apecloud/kubeblocks/apis/apps/v1alpha1"
+ workloads "github.com/apecloud/kubeblocks/apis/workloads/v1alpha1"
"github.com/apecloud/kubeblocks/internal/configuration/core"
intctrlutil "github.com/apecloud/kubeblocks/internal/controllerutil"
)
@@ -91,6 +92,9 @@ func newMockStatefulSet(replicas int, name string, labels map[string]string) app
UID: types.UID(uid),
},
Spec: appsv1.StatefulSetSpec{
+ Selector: &metav1.LabelSelector{
+ MatchLabels: labels,
+ },
Replicas: func() *int32 { i := int32(replicas); return &i }(),
Template: corev1.PodTemplateSpec{
ObjectMeta: metav1.ObjectMeta{
@@ -199,14 +203,20 @@ func withCDComponent(compType appsv1alpha1.WorkloadType, tpls []appsv1alpha1.Com
WorkloadType: compType,
Name: string(compType),
}
- if compType == appsv1alpha1.Consensus {
- params.Component.ConsensusSpec = &appsv1alpha1.ConsensusSetSpec{
- Leader: appsv1alpha1.ConsensusMember{
- Name: "leader",
- },
- Followers: []appsv1alpha1.ConsensusMember{
+ if compType == appsv1alpha1.Consensus || compType == appsv1alpha1.Replication {
+ params.Component.RSMSpec = &appsv1alpha1.RSMSpec{
+ Roles: []workloads.ReplicaRole{
+ {
+ Name: "leader",
+ IsLeader: true,
+ AccessMode: workloads.ReadWriteMode,
+ CanVote: true,
+ },
{
- Name: "follower",
+ Name: "follower",
+ IsLeader: false,
+ AccessMode: workloads.ReadonlyMode,
+ CanVote: true,
},
},
}
diff --git a/controllers/apps/configuration/reconcile_task.go b/controllers/apps/configuration/reconcile_task.go
index 874cdc28096..03f1510682a 100644
--- a/controllers/apps/configuration/reconcile_task.go
+++ b/controllers/apps/configuration/reconcile_task.go
@@ -22,6 +22,8 @@ package configuration
import (
"strconv"
+ corev1 "k8s.io/api/core/v1"
+
appsv1alpha1 "github.com/apecloud/kubeblocks/apis/apps/v1alpha1"
"github.com/apecloud/kubeblocks/internal/configuration/core"
"github.com/apecloud/kubeblocks/internal/controller/component"
@@ -35,47 +37,73 @@ type Task struct {
Status *appsv1alpha1.ConfigurationItemDetailStatus
Name string
- Do func(fetcher *Task, component *component.SynthesizedComponent, revision string) error
- SyncStatus func(fetcher *Task, status *appsv1alpha1.ConfigurationItemDetailStatus) error
+ Do func(fetcher *Task, component *component.SynthesizedComponent, revision string) error
+}
+
+type TaskContext struct {
+ configuration *appsv1alpha1.Configuration
+ reqCtx intctrlutil.RequestCtx
+ fetcher *Task
}
func NewTask(item appsv1alpha1.ConfigurationItemDetail, status *appsv1alpha1.ConfigurationItemDetailStatus) Task {
return Task{
- Name: item.Name,
- Status: status,
+ Name: item.Name,
Do: func(fetcher *Task, synComponent *component.SynthesizedComponent, revision string) error {
- configSpec := component.GetConfigSpecByName(synComponent, item.Name)
+ configSpec := item.ConfigSpec
if configSpec == nil {
return core.MakeError("not found config spec: %s", item.Name)
}
- reconcileTask := configuration.NewReconcilePipeline(configuration.ReconcileCtx{
- ResourceCtx: fetcher.ResourceCtx,
- Cluster: fetcher.ClusterObj,
- ClusterVer: fetcher.ClusterVerObj,
- Component: synComponent,
- PodSpec: synComponent.PodSpec,
- }, item, status, configSpec)
- return reconcileTask.ConfigMap(item.Name).
- ConfigConstraints(configSpec.ConfigConstraintRef).
- PrepareForTemplate().
- RerenderTemplate().
- ApplyParameters().
- UpdateConfigVersion(revision).
- Sync().
- SyncStatus().
- Complete()
+ if err := fetcher.ConfigMap(item.Name).Complete(); err != nil {
+ return err
+ }
+ // Do reconcile for config template
+ configMap := fetcher.ConfigMapObj
+ switch intctrlutil.GetConfigSpecReconcilePhase(configMap, item, status) {
+ default:
+ return syncStatus(configMap, status)
+ case appsv1alpha1.CPendingPhase,
+ appsv1alpha1.CMergeFailedPhase:
+ return syncImpl(fetcher, item, status, synComponent, revision, configSpec)
+ case appsv1alpha1.CCreatingPhase:
+ return nil
+ }
},
- SyncStatus: syncStatus,
+ Status: status,
}
}
-func syncStatus(fetcher *Task, status *appsv1alpha1.ConfigurationItemDetailStatus) (err error) {
- err = fetcher.ConfigMap(status.Name).Complete()
+func syncImpl(fetcher *Task,
+ item appsv1alpha1.ConfigurationItemDetail,
+ status *appsv1alpha1.ConfigurationItemDetailStatus,
+ component *component.SynthesizedComponent,
+ revision string,
+ configSpec *appsv1alpha1.ComponentConfigSpec) (err error) {
+ err = configuration.NewReconcilePipeline(configuration.ReconcileCtx{
+ ResourceCtx: fetcher.ResourceCtx,
+ Cluster: fetcher.ClusterObj,
+ ClusterVer: fetcher.ClusterVerObj,
+ Component: component,
+ PodSpec: component.PodSpec,
+ }, item, status, configSpec).
+ ConfigMap(item.Name).
+ ConfigConstraints(configSpec.ConfigConstraintRef).
+ PrepareForTemplate().
+ RerenderTemplate().
+ ApplyParameters().
+ UpdateConfigVersion(revision).
+ Sync().
+ Complete()
if err != nil {
- return
+ status.Phase = appsv1alpha1.CMergeFailedPhase
+ } else {
+ status.Phase = appsv1alpha1.CMergedPhase
}
+ return
+}
- annotations := fetcher.ConfigMapObj.GetAnnotations()
+func syncStatus(configMap *corev1.ConfigMap, status *appsv1alpha1.ConfigurationItemDetailStatus) (err error) {
+ annotations := configMap.GetAnnotations()
// status.CurrentRevision = GetCurrentRevision(annotations)
revisions := RetrieveRevision(annotations)
if len(revisions) == 0 {
diff --git a/controllers/apps/configuration/revision.go b/controllers/apps/configuration/revision.go
index 7fabec72ad8..3af84e14b70 100644
--- a/controllers/apps/configuration/revision.go
+++ b/controllers/apps/configuration/revision.go
@@ -24,10 +24,11 @@ import (
"strconv"
"strings"
+ corev1 "k8s.io/api/core/v1"
+
appsv1alpha1 "github.com/apecloud/kubeblocks/apis/apps/v1alpha1"
"github.com/apecloud/kubeblocks/internal/configuration/core"
"github.com/apecloud/kubeblocks/internal/constant"
- corev1 "k8s.io/api/core/v1"
)
type ConfigurationRevision struct {
diff --git a/controllers/apps/configuration/revision_test.go b/controllers/apps/configuration/revision_test.go
index 49a351da841..428c1b2126d 100644
--- a/controllers/apps/configuration/revision_test.go
+++ b/controllers/apps/configuration/revision_test.go
@@ -23,11 +23,11 @@ import (
"fmt"
"testing"
- "github.com/apecloud/kubeblocks/internal/constant"
"github.com/stretchr/testify/assert"
appsv1alpha1 "github.com/apecloud/kubeblocks/apis/apps/v1alpha1"
"github.com/apecloud/kubeblocks/internal/configuration/core"
+ "github.com/apecloud/kubeblocks/internal/constant"
"github.com/apecloud/kubeblocks/internal/controller/builder"
)
diff --git a/controllers/apps/configuration/rolling_upgrade_policy.go b/controllers/apps/configuration/rolling_upgrade_policy.go
index 14c4ba8d8e2..356519f5365 100644
--- a/controllers/apps/configuration/rolling_upgrade_policy.go
+++ b/controllers/apps/configuration/rolling_upgrade_policy.go
@@ -53,18 +53,17 @@ func init() {
}
func (r *rollingUpgradePolicy) Upgrade(params reconfigureParams) (ReturnedStatus, error) {
- var (
- funcs RollingUpgradeFuncs
- cType = params.WorkloadType()
- )
+ var funcs RollingUpgradeFuncs
- switch cType {
+ switch params.WorkloadType() {
case appsv1alpha1.Consensus:
funcs = GetConsensusRollingUpgradeFuncs()
+ case appsv1alpha1.Replication:
+ funcs = GetReplicationRollingUpgradeFuncs()
case appsv1alpha1.Stateful:
funcs = GetStatefulSetRollingUpgradeFuncs()
default:
- return makeReturnedStatus(ESNotSupport), cfgcore.MakeError("not supported component workload type[%s]", cType)
+ return makeReturnedStatus(ESNotSupport), cfgcore.MakeError("not supported component workload type[%s]", params.WorkloadType())
}
return performRollingUpgrade(params, funcs)
}
diff --git a/controllers/apps/configuration/types.go b/controllers/apps/configuration/types.go
index 4afaa420afc..32276a6ff3b 100644
--- a/controllers/apps/configuration/types.go
+++ b/controllers/apps/configuration/types.go
@@ -37,6 +37,9 @@ type RestartComponent func(client client.Client, ctx intctrlutil.RequestCtx, key
type RestartContainerFunc func(pod *corev1.Pod, ctx context.Context, containerName []string, createConnFn createReconfigureClient) error
type OnlineUpdatePodFunc func(pod *corev1.Pod, ctx context.Context, createClient createReconfigureClient, configSpec string, updatedParams map[string]string) error
+// Node: Distinguish between implementation and interface.
+// RollingUpgradeFuncs defines the interface, rsm is an implementation of Stateful, Replication and Consensus, not the only solution.
+
type RollingUpgradeFuncs struct {
GetPodsFunc GetPodsFunc
RestartContainerFunc RestartContainerFunc
diff --git a/controllers/apps/operations/backup.go b/controllers/apps/operations/backup.go
index 468a35432a0..f8f05112d38 100644
--- a/controllers/apps/operations/backup.go
+++ b/controllers/apps/operations/backup.go
@@ -28,9 +28,10 @@ import (
"sigs.k8s.io/controller-runtime/pkg/client"
appsv1alpha1 "github.com/apecloud/kubeblocks/apis/apps/v1alpha1"
- dataprotectionv1alpha1 "github.com/apecloud/kubeblocks/apis/dataprotection/v1alpha1"
+ dpv1alpha1 "github.com/apecloud/kubeblocks/apis/dataprotection/v1alpha1"
"github.com/apecloud/kubeblocks/internal/constant"
intctrlutil "github.com/apecloud/kubeblocks/internal/controllerutil"
+ dptypes "github.com/apecloud/kubeblocks/internal/dataprotection/types"
)
const backupTimeLayout = "20060102150405"
@@ -79,7 +80,7 @@ func (b BackupOpsHandler) ReconcileAction(reqCtx intctrlutil.RequestCtx, cli cli
cluster := opsRes.Cluster
// get backup
- backups := &dataprotectionv1alpha1.BackupList{}
+ backups := &dpv1alpha1.BackupList{}
if err := cli.List(reqCtx.Ctx, backups, client.InNamespace(cluster.Namespace), client.MatchingLabels(getBackupLabels(cluster.Name, opsRequest.Name))); err != nil {
return appsv1alpha1.OpsFailedPhase, 0, err
}
@@ -89,9 +90,9 @@ func (b BackupOpsHandler) ReconcileAction(reqCtx intctrlutil.RequestCtx, cli cli
}
// check backup status
phase := backups.Items[0].Status.Phase
- if phase == dataprotectionv1alpha1.BackupCompleted {
+ if phase == dpv1alpha1.BackupPhaseCompleted {
return appsv1alpha1.OpsSucceedPhase, 0, nil
- } else if phase == dataprotectionv1alpha1.BackupFailed {
+ } else if phase == dpv1alpha1.BackupPhaseFailed {
return appsv1alpha1.OpsFailedPhase, 0, fmt.Errorf("backup failed")
}
return appsv1alpha1.OpsRunningPhase, 0, nil
@@ -102,14 +103,12 @@ func (b BackupOpsHandler) SaveLastConfiguration(reqCtx intctrlutil.RequestCtx, c
return nil
}
-func buildBackup(reqCtx intctrlutil.RequestCtx, cli client.Client, opsRequest *appsv1alpha1.OpsRequest, cluster *appsv1alpha1.Cluster) (*dataprotectionv1alpha1.Backup, error) {
+func buildBackup(reqCtx intctrlutil.RequestCtx, cli client.Client, opsRequest *appsv1alpha1.OpsRequest, cluster *appsv1alpha1.Cluster) (*dpv1alpha1.Backup, error) {
var err error
backupSpec := opsRequest.Spec.BackupSpec
if backupSpec == nil {
- backupSpec = &appsv1alpha1.BackupSpec{
- BackupType: string(dataprotectionv1alpha1.BackupTypeDataFile),
- }
+ backupSpec = &appsv1alpha1.BackupSpec{}
}
if len(backupSpec.BackupName) == 0 {
@@ -121,15 +120,15 @@ func buildBackup(reqCtx intctrlutil.RequestCtx, cli client.Client, opsRequest *a
return nil, err
}
- backup := &dataprotectionv1alpha1.Backup{
+ backup := &dpv1alpha1.Backup{
ObjectMeta: metav1.ObjectMeta{
Name: backupSpec.BackupName,
Namespace: cluster.Namespace,
Labels: getBackupLabels(cluster.Name, opsRequest.Name),
},
- Spec: dataprotectionv1alpha1.BackupSpec{
+ Spec: dpv1alpha1.BackupSpec{
BackupPolicyName: backupSpec.BackupPolicyName,
- BackupType: dataprotectionv1alpha1.BackupType(backupSpec.BackupType),
+ BackupMethod: backupSpec.BackupMethod,
},
}
@@ -142,28 +141,28 @@ func getDefaultBackupPolicy(reqCtx intctrlutil.RequestCtx, cli client.Client, cl
return backupPolicy, nil
}
- backupPolicyList := &dataprotectionv1alpha1.BackupPolicyList{}
+ backupPolicyList := &dpv1alpha1.BackupPolicyList{}
if err := cli.List(reqCtx.Ctx, backupPolicyList, client.InNamespace(cluster.Namespace),
client.MatchingLabels(map[string]string{
constant.AppInstanceLabelKey: cluster.Name,
})); err != nil {
return "", err
}
- defaultBackupPolicys := &dataprotectionv1alpha1.BackupPolicyList{}
+ defaultBackupPolices := &dpv1alpha1.BackupPolicyList{}
for _, backupPolicy := range backupPolicyList.Items {
- if backupPolicy.GetAnnotations()[constant.DefaultBackupPolicyAnnotationKey] == "true" {
- defaultBackupPolicys.Items = append(defaultBackupPolicys.Items, backupPolicy)
+ if backupPolicy.GetAnnotations()[dptypes.DefaultBackupPolicyAnnotationKey] == "true" {
+ defaultBackupPolices.Items = append(defaultBackupPolices.Items, backupPolicy)
}
}
- if len(defaultBackupPolicys.Items) == 0 {
+ if len(defaultBackupPolices.Items) == 0 {
return "", fmt.Errorf(`not found any default backup policy for cluster "%s"`, cluster.Name)
}
- if len(defaultBackupPolicys.Items) > 1 {
+ if len(defaultBackupPolices.Items) > 1 {
return "", fmt.Errorf(`cluster "%s" has multiple default backup policies`, cluster.Name)
}
- return defaultBackupPolicys.Items[0].GetName(), nil
+ return defaultBackupPolices.Items[0].GetName(), nil
}
func getBackupLabels(cluster, request string) map[string]string {
diff --git a/controllers/apps/operations/datascript.go b/controllers/apps/operations/datascript.go
index 1980521de12..c44f3518a82 100644
--- a/controllers/apps/operations/datascript.go
+++ b/controllers/apps/operations/datascript.go
@@ -24,6 +24,8 @@ import (
"strings"
"time"
+ "github.com/sethvargo/go-password/password"
+
batchv1 "k8s.io/api/batch/v1"
corev1 "k8s.io/api/core/v1"
apierrors "k8s.io/apimachinery/pkg/api/errors"
@@ -97,11 +99,16 @@ func (o DataScriptOpsHandler) Action(reqCtx intctrlutil.RequestCtx, cli client.C
}
// create jobs
- if job, err := buildDataScriptJob(reqCtx, cli, opsResource.Cluster, component, opsRequest, componentDef.CharacterType); err != nil {
+ var jobs []*batchv1.Job
+ if jobs, err = buildDataScriptJobs(reqCtx, cli, opsResource.Cluster, component, opsRequest, componentDef.CharacterType); err != nil {
return err
- } else {
- return cli.Create(reqCtx.Ctx, job)
}
+ for _, job := range jobs {
+ if err = cli.Create(reqCtx.Ctx, job); err != nil {
+ return err
+ }
+ }
+ return nil
}
// ReconcileAction implements OpsHandler.ReconcileAction
@@ -114,37 +121,60 @@ func (o DataScriptOpsHandler) ReconcileAction(reqCtx intctrlutil.RequestCtx, cli
cluster := opsResource.Cluster
spec := opsRequest.Spec.ScriptSpec
- getStatusFromJobCondition := func(job *batchv1.Job) appsv1alpha1.OpsPhase {
+ meetsJobConditions := func(job *batchv1.Job, condType batchv1.JobConditionType, condStatus corev1.ConditionStatus) bool {
for _, condition := range job.Status.Conditions {
- if condition.Type == batchv1.JobComplete && condition.Status == corev1.ConditionTrue {
- return appsv1alpha1.OpsSucceedPhase
- } else if condition.Type == batchv1.JobFailed && condition.Status == corev1.ConditionTrue {
- return appsv1alpha1.OpsFailedPhase
+ if condition.Type == condType && condition.Status == condStatus {
+ return true
}
}
- return appsv1alpha1.OpsRunningPhase
+ return false
}
// retrieve job for this opsRequest
jobList := &batchv1.JobList{}
- err := cli.List(reqCtx.Ctx, jobList, client.InNamespace(cluster.Namespace), client.MatchingLabels(getDataScriptJobLabels(cluster.Name, spec.ComponentName, opsRequest.Name)))
- if err != nil {
+ if err := cli.List(reqCtx.Ctx, jobList, client.InNamespace(cluster.Namespace), client.MatchingLabels(getDataScriptJobLabels(cluster.Name, spec.ComponentName, opsRequest.Name))); err != nil {
return appsv1alpha1.OpsFailedPhase, 0, err
- }
-
- if len(jobList.Items) == 0 {
+ } else if len(jobList.Items) == 0 {
return appsv1alpha1.OpsFailedPhase, 0, fmt.Errorf("job not found")
}
+
+ var (
+ expectedCount int
+ succedCount int
+ failedCount int
+ )
+
+ expectedCount = len(jobList.Items)
// check job status
- job := &jobList.Items[0]
- phase := getStatusFromJobCondition(job)
- // jobs are owned by opsRequest, so we don't need to delete them explicitly
- if phase == appsv1alpha1.OpsFailedPhase {
- return phase, 0, fmt.Errorf("job execution failed, please check the job log with `kubectl logs jobs/%s -n %s`", job.Name, job.Namespace)
- } else if phase == appsv1alpha1.OpsSucceedPhase {
- return phase, 0, nil
+ for _, job := range jobList.Items {
+ if meetsJobConditions(&job, batchv1.JobComplete, corev1.ConditionTrue) {
+ succedCount++
+ } else if meetsJobConditions(&job, batchv1.JobFailed, corev1.ConditionTrue) {
+ failedCount++
+ }
+ }
+
+ opsStatus := appsv1alpha1.OpsRunningPhase
+ if succedCount == expectedCount {
+ opsStatus = appsv1alpha1.OpsSucceedPhase
+ } else if failedCount+succedCount == expectedCount {
+ opsStatus = appsv1alpha1.OpsFailedPhase
}
- return phase, time.Second, nil
+
+ patch := client.MergeFrom(opsRequest.DeepCopy())
+ opsRequest.Status.Progress = fmt.Sprintf("%d/%d", succedCount, expectedCount)
+
+ // patch OpsRequest.status.components
+ if err := cli.Status().Patch(reqCtx.Ctx, opsRequest, patch); err != nil {
+ return opsStatus, time.Second, err
+ }
+
+ if succedCount == expectedCount {
+ return appsv1alpha1.OpsSucceedPhase, 0, nil
+ } else if failedCount+succedCount == expectedCount {
+ return appsv1alpha1.OpsFailedPhase, 0, fmt.Errorf("%d job execution failed, please check the job log ", failedCount)
+ }
+ return appsv1alpha1.OpsRunningPhase, 5 * time.Second, nil
}
func (o DataScriptOpsHandler) ActionStartedCondition(reqCtx intctrlutil.RequestCtx, cli client.Client, opsRes *OpsResource) (*metav1.Condition, error) {
@@ -207,127 +237,168 @@ func getTargetService(reqCtx intctrlutil.RequestCtx, cli client.Client, clusterO
return serviceName, nil
}
-func buildDataScriptJob(reqCtx intctrlutil.RequestCtx, cli client.Client, cluster *appsv1alpha1.Cluster, component *appsv1alpha1.ClusterComponentSpec,
- ops *appsv1alpha1.OpsRequest, charType string) (*batchv1.Job, error) {
+func buildDataScriptJobs(reqCtx intctrlutil.RequestCtx, cli client.Client, cluster *appsv1alpha1.Cluster, component *appsv1alpha1.ClusterComponentSpec,
+ ops *appsv1alpha1.OpsRequest, charType string) ([]*batchv1.Job, error) {
engineForJob, err := engine.New(charType)
if err != nil || engineForJob == nil {
return nil, &FastFaileError{message: err.Error()}
}
- envs := []corev1.EnvVar{}
- // parse kb host
- serviceName, err := getTargetService(reqCtx, cli, client.ObjectKeyFromObject(cluster), component.Name)
- if err != nil {
- return nil, &FastFaileError{message: err.Error()}
- }
-
- envs = append(envs, corev1.EnvVar{
- Name: "KB_HOST",
- Value: serviceName,
- })
-
- // parse username and password
- secretFrom := ops.Spec.ScriptSpec.Secret
- if secretFrom == nil {
- secretFrom = &appsv1alpha1.ScriptSecret{
- Name: fmt.Sprintf("%s-conn-credential", cluster.Name),
- PasswordKey: "password",
- UsernameKey: "username",
+ buildJob := func(endpoint string) (*batchv1.Job, error) {
+ envs := []corev1.EnvVar{}
+
+ envs = append(envs, corev1.EnvVar{
+ Name: "KB_HOST",
+ Value: endpoint,
+ })
+
+ // parse username and password
+ secretFrom := ops.Spec.ScriptSpec.Secret
+ if secretFrom == nil {
+ secretFrom = &appsv1alpha1.ScriptSecret{
+ Name: fmt.Sprintf("%s-conn-credential", cluster.Name),
+ PasswordKey: "password",
+ UsernameKey: "username",
+ }
+ }
+ // verify secrets exist
+ if err := cli.Get(reqCtx.Ctx, types.NamespacedName{Namespace: reqCtx.Req.Namespace, Name: secretFrom.Name}, &corev1.Secret{}); err != nil {
+ return nil, &FastFaileError{message: err.Error()}
}
- }
- // verify secrets exist
- if err := cli.Get(reqCtx.Ctx, types.NamespacedName{Namespace: reqCtx.Req.Namespace, Name: secretFrom.Name}, &corev1.Secret{}); err != nil {
- return nil, &FastFaileError{message: err.Error()}
- }
- envs = append(envs, corev1.EnvVar{
- Name: "KB_USER",
- ValueFrom: &corev1.EnvVarSource{
- SecretKeyRef: &corev1.SecretKeySelector{
- Key: secretFrom.UsernameKey,
- LocalObjectReference: corev1.LocalObjectReference{
- Name: secretFrom.Name,
+ envs = append(envs, corev1.EnvVar{
+ Name: "KB_USER",
+ ValueFrom: &corev1.EnvVarSource{
+ SecretKeyRef: &corev1.SecretKeySelector{
+ Key: secretFrom.UsernameKey,
+ LocalObjectReference: corev1.LocalObjectReference{
+ Name: secretFrom.Name,
+ },
},
},
- },
- })
- envs = append(envs, corev1.EnvVar{
- Name: "KB_PASSWD",
- ValueFrom: &corev1.EnvVarSource{
- SecretKeyRef: &corev1.SecretKeySelector{
- Key: secretFrom.PasswordKey,
- LocalObjectReference: corev1.LocalObjectReference{
- Name: secretFrom.Name,
+ })
+ envs = append(envs, corev1.EnvVar{
+ Name: "KB_PASSWD",
+ ValueFrom: &corev1.EnvVarSource{
+ SecretKeyRef: &corev1.SecretKeySelector{
+ Key: secretFrom.PasswordKey,
+ LocalObjectReference: corev1.LocalObjectReference{
+ Name: secretFrom.Name,
+ },
},
},
- },
- })
+ })
- // parse scripts
- scripts, err := getScriptContent(reqCtx, cli, ops.Spec.ScriptSpec)
- if err != nil {
- return nil, &FastFaileError{message: err.Error()}
- }
+ // parse scripts
+ scripts, err := getScriptContent(reqCtx, cli, ops.Spec.ScriptSpec)
+ if err != nil {
+ return nil, &FastFaileError{message: err.Error()}
+ }
- envs = append(envs, corev1.EnvVar{
- Name: "KB_SCRIPT",
- Value: strings.Join(scripts, "\n"),
- })
+ envs = append(envs, corev1.EnvVar{
+ Name: "KB_SCRIPT",
+ Value: strings.Join(scripts, "\n"),
+ })
- jobCmdTpl, envVars, err := engineForJob.ExecuteCommand(scripts)
- if err != nil {
- return nil, &FastFaileError{message: err.Error()}
- }
- if envVars != nil {
- envs = append(envs, envVars...)
- }
- containerImg := viper.GetString(constant.KBDataScriptClientsImage)
- if len(ops.Spec.ScriptSpec.Image) != 0 {
- containerImg = ops.Spec.ScriptSpec.Image
- }
- if len(containerImg) == 0 {
- return nil, &FastFaileError{message: "image is empty"}
- }
+ jobCmdTpl, envVars, err := engineForJob.ExecuteCommand(scripts)
+ if err != nil {
+ return nil, &FastFaileError{message: err.Error()}
+ }
+ if envVars != nil {
+ envs = append(envs, envVars...)
+ }
+ containerImg := viper.GetString(constant.KBDataScriptClientsImage)
+ if len(ops.Spec.ScriptSpec.Image) != 0 {
+ containerImg = ops.Spec.ScriptSpec.Image
+ }
+ if len(containerImg) == 0 {
+ return nil, &FastFaileError{message: "image is empty"}
+ }
- container := corev1.Container{
- Name: "datascript",
- Image: containerImg,
- ImagePullPolicy: corev1.PullPolicy(viper.GetString(constant.KBImagePullPolicy)),
- Command: jobCmdTpl,
- Env: envs,
- }
+ container := corev1.Container{
+ Name: "datascript",
+ Image: containerImg,
+ ImagePullPolicy: corev1.PullPolicy(viper.GetString(constant.KBImagePullPolicy)),
+ Command: jobCmdTpl,
+ Env: envs,
+ }
+ randomStr, _ := password.Generate(4, 0, 0, true, false)
+ jobName := fmt.Sprintf("%s-%s-%s-%s", cluster.Name, "script", ops.Name, randomStr)
+ if len(jobName) > 63 {
+ jobName = jobName[:63]
+ }
- jobName := fmt.Sprintf("%s-%s-%s", cluster.Name, "script", ops.Name)
- if len(jobName) > 63 {
- jobName = jobName[:63]
- }
+ job := &batchv1.Job{
+ ObjectMeta: metav1.ObjectMeta{
+ Name: jobName,
+ Namespace: cluster.Namespace,
+ },
+ }
- job := &batchv1.Job{
- ObjectMeta: metav1.ObjectMeta{
- Name: jobName,
- Namespace: cluster.Namespace,
- },
+ // set backoff limit to 0, so that the job will not be restarted
+ job.Spec.BackoffLimit = pointer.Int32(0)
+ job.Spec.Template.Spec.RestartPolicy = corev1.RestartPolicyNever
+ job.Spec.Template.Spec.Containers = []corev1.Container{container}
+
+ // add labels
+ job.Labels = getDataScriptJobLabels(cluster.Name, component.Name, ops.Name)
+ // add tolerations
+ tolerations, err := componetutil.BuildTolerations(cluster, component)
+ if err != nil {
+ return nil, &FastFaileError{message: err.Error()}
+ }
+ job.Spec.Template.Spec.Tolerations = tolerations
+ // add owner reference
+ scheme, _ := appsv1alpha1.SchemeBuilder.Build()
+ if err := controllerutil.SetOwnerReference(ops, job, scheme); err != nil {
+ return nil, &FastFaileError{message: err.Error()}
+ }
+ return job, nil
}
- // set backoff limit to 0, so that the job will not be restarted
- job.Spec.BackoffLimit = pointer.Int32Ptr(0)
- job.Spec.Template.Spec.RestartPolicy = corev1.RestartPolicyNever
- job.Spec.Template.Spec.Containers = []corev1.Container{container}
+ // parse kb host
+ var endpoint string
+ var job *batchv1.Job
- // add labels
- job.Labels = getDataScriptJobLabels(cluster.Name, component.Name, ops.Name)
- // add tolerations
- tolerations, err := componetutil.BuildTolerations(cluster, component)
+ jobs := make([]*batchv1.Job, 0)
+ if ops.Spec.ScriptSpec.Selector == nil {
+ if endpoint, err = getTargetService(reqCtx, cli, client.ObjectKeyFromObject(cluster), component.Name); err != nil {
+ return nil, &FastFaileError{message: err.Error()}
+ }
+ if job, err = buildJob(endpoint); err != nil {
+ return nil, &FastFaileError{message: err.Error()}
+ }
+ jobs = append(jobs, job)
+ return jobs, nil
+ }
+
+ selector, err := metav1.LabelSelectorAsSelector(ops.Spec.ScriptSpec.Selector)
if err != nil {
return nil, &FastFaileError{message: err.Error()}
}
- job.Spec.Template.Spec.Tolerations = tolerations
- // add owner reference
- scheme, _ := appsv1alpha1.SchemeBuilder.Build()
- if err := controllerutil.SetOwnerReference(ops, job, scheme); err != nil {
+
+ pods := &corev1.PodList{}
+ if err = cli.List(reqCtx.Ctx, pods, client.InNamespace(cluster.Namespace),
+ client.MatchingLabels{
+ constant.AppInstanceLabelKey: cluster.Name,
+ constant.KBAppComponentLabelKey: component.Name,
+ },
+ client.MatchingLabelsSelector{Selector: selector},
+ ); err != nil {
return nil, &FastFaileError{message: err.Error()}
+ } else if len(pods.Items) == 0 {
+ return nil, &FastFaileError{message: "no pods found"}
+ }
+
+ for _, pod := range pods.Items {
+ endpoint = pod.Status.PodIP
+ if job, err = buildJob(endpoint); err != nil {
+ return nil, &FastFaileError{message: err.Error()}
+ } else {
+ jobs = append(jobs, job)
+ }
}
- return job, nil
+ return jobs, nil
}
func getDataScriptJobLabels(cluster, component, request string) map[string]string {
diff --git a/controllers/apps/operations/datascript_test.go b/controllers/apps/operations/datascript_test.go
index 3a46c1568d9..5e477d7d614 100644
--- a/controllers/apps/operations/datascript_test.go
+++ b/controllers/apps/operations/datascript_test.go
@@ -236,7 +236,7 @@ var _ = Describe("DataScriptOps", func() {
reqCtx.Req = reconcile.Request{NamespacedName: opsKey}
By("mock a job, missing service, should fail")
comp := clusterObj.Spec.GetComponentByName(consensusComp)
- _, err := buildDataScriptJob(reqCtx, k8sClient, clusterObj, comp, ops, "mysql")
+ _, err := buildDataScriptJobs(reqCtx, k8sClient, clusterObj, comp, ops, "mysql")
Expect(err).Should(HaveOccurred())
By("mock a service, should pass")
@@ -249,7 +249,7 @@ var _ = Describe("DataScriptOps", func() {
Expect(err).Should(Succeed())
By("mock a job one more time, fail with missing secret")
- _, err = buildDataScriptJob(reqCtx, k8sClient, clusterObj, comp, ops, "mysql")
+ _, err = buildDataScriptJobs(reqCtx, k8sClient, clusterObj, comp, ops, "mysql")
Expect(err).Should(HaveOccurred())
Expect(err.Error()).Should(ContainSubstring("conn-credential"))
@@ -263,7 +263,7 @@ var _ = Describe("DataScriptOps", func() {
}
Expect(k8sClient.Patch(testCtx.Ctx, ops, patch)).Should(Succeed())
- _, err = buildDataScriptJob(reqCtx, k8sClient, clusterObj, comp, ops, "mysql")
+ _, err = buildDataScriptJobs(reqCtx, k8sClient, clusterObj, comp, ops, "mysql")
Expect(err).Should(HaveOccurred())
Expect(err.Error()).Should(ContainSubstring(secretName))
@@ -281,8 +281,9 @@ var _ = Describe("DataScriptOps", func() {
By("create job, should pass")
viper.Set(constant.KBDataScriptClientsImage, "apecloud/kubeblocks-clients:latest")
- job, err := buildDataScriptJob(reqCtx, k8sClient, clusterObj, comp, ops, "mysql")
+ jobs, err := buildDataScriptJobs(reqCtx, k8sClient, clusterObj, comp, ops, "mysql")
Expect(err).Should(Succeed())
+ job := jobs[0]
Expect(k8sClient.Create(testCtx.Ctx, job)).Should(Succeed())
By("reconcile the opsRequest phase")
@@ -358,8 +359,9 @@ var _ = Describe("DataScriptOps", func() {
By("create job, should pass")
viper.Set(constant.KBDataScriptClientsImage, "apecloud/kubeblocks-clients:latest")
- job, err := buildDataScriptJob(reqCtx, k8sClient, clusterObj, comp, ops, "mysql")
+ jobs, err := buildDataScriptJobs(reqCtx, k8sClient, clusterObj, comp, ops, "mysql")
Expect(err).Should(Succeed())
+ job := jobs[0]
Expect(k8sClient.Create(testCtx.Ctx, job)).Should(Succeed())
By("reconcile the opsRequest phase")
diff --git a/controllers/apps/operations/ops_progress_util.go b/controllers/apps/operations/ops_progress_util.go
index 0ff1112d2a7..d5fd0b3922d 100644
--- a/controllers/apps/operations/ops_progress_util.go
+++ b/controllers/apps/operations/ops_progress_util.go
@@ -27,6 +27,7 @@ import (
corev1 "k8s.io/api/core/v1"
metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
"k8s.io/client-go/tools/record"
+ "k8s.io/kubectl/pkg/util/podutils"
"sigs.k8s.io/controller-runtime/pkg/client"
appsv1alpha1 "github.com/apecloud/kubeblocks/apis/apps/v1alpha1"
@@ -292,7 +293,7 @@ func handleCancelProgressForPodsRollingUpdate(
objectKey := getProgressObjectKey(pod.Kind, pod.Name)
progressDetail := appsv1alpha1.ProgressStatusDetail{ObjectKey: objectKey}
if !pod.CreationTimestamp.Before(&opsCancelTime) &&
- components.PodIsAvailable(workloadType, &pod, minReadySeconds) {
+ podIsAvailable(workloadType, &pod, minReadySeconds) {
completedCount += 1
handleSucceedProgressDetail(opsRes, pgRes, compStatus, progressDetail)
continue
@@ -305,6 +306,20 @@ func handleCancelProgressForPodsRollingUpdate(
return completedCount
}
+func podIsAvailable(workloadType appsv1alpha1.WorkloadType, pod *corev1.Pod, minReadySeconds int32) bool {
+ if pod == nil {
+ return false
+ }
+ switch workloadType {
+ case appsv1alpha1.Consensus, appsv1alpha1.Replication:
+ return intctrlutil.PodIsReadyWithLabel(*pod)
+ case appsv1alpha1.Stateful, appsv1alpha1.Stateless:
+ return podutils.IsPodAvailable(pod, minReadySeconds, metav1.Time{Time: time.Now()})
+ default:
+ panic("unknown workload type")
+ }
+}
+
// handlePendingProgressDetail handles the pending progressDetail and sets it to progressDetails.
func handlePendingProgressDetail(opsRes *OpsResource,
compStatus *appsv1alpha1.OpsRequestComponentStatus,
@@ -379,7 +394,7 @@ func podProcessedSuccessful(workloadType appsv1alpha1.WorkloadType,
minReadySeconds int32,
componentPhase appsv1alpha1.ClusterComponentPhase,
opsIsCompleted bool) bool {
- if !components.PodIsAvailable(workloadType, pod, minReadySeconds) {
+ if !podIsAvailable(workloadType, pod, minReadySeconds) {
return false
}
return (opsIsCompleted && componentPhase == appsv1alpha1.RunningClusterCompPhase) || !pod.CreationTimestamp.Before(&opsStartTime)
@@ -506,7 +521,7 @@ func handleScaleOutProgress(reqCtx intctrlutil.RequestCtx,
objectKey := getProgressObjectKey(v.Kind, v.Name)
progressDetail := appsv1alpha1.ProgressStatusDetail{ObjectKey: objectKey}
pgRes.opsMessageKey = "create"
- if components.PodIsAvailable(workloadType, &v, minReadySeconds) {
+ if podIsAvailable(workloadType, &v, minReadySeconds) {
completedCount += 1
handleSucceedProgressDetail(opsRes, pgRes, compStatus, progressDetail)
continue
@@ -572,7 +587,7 @@ func handleScaleDownProgress(
}
// handle the re-created pods if these pods are failed before doing horizontal scaling.
pgRes.opsMessageKey = "re-create"
- if components.PodIsAvailable(workloadType, &pod, minReadySeconds) {
+ if podIsAvailable(workloadType, &pod, minReadySeconds) {
completedCount += 1
handleSucceedProgressDetail(opsRes, pgRes, compStatus, progressDetail)
continue
diff --git a/controllers/apps/operations/pipeline.go b/controllers/apps/operations/pipeline.go
index f1bd1251e03..c0da1cd6912 100644
--- a/controllers/apps/operations/pipeline.go
+++ b/controllers/apps/operations/pipeline.go
@@ -20,12 +20,12 @@ along with this program. If not, see .
package operations
import (
- "github.com/apecloud/kubeblocks/internal/controller/configuration"
"sigs.k8s.io/controller-runtime/pkg/client"
appsv1alpha1 "github.com/apecloud/kubeblocks/apis/apps/v1alpha1"
cfgcore "github.com/apecloud/kubeblocks/internal/configuration/core"
"github.com/apecloud/kubeblocks/internal/configuration/validate"
+ "github.com/apecloud/kubeblocks/internal/controller/configuration"
intctrlutil "github.com/apecloud/kubeblocks/internal/controllerutil"
)
diff --git a/controllers/apps/operations/reconfigure.go b/controllers/apps/operations/reconfigure.go
index c0a978c6070..a88ea5c3872 100644
--- a/controllers/apps/operations/reconfigure.go
+++ b/controllers/apps/operations/reconfigure.go
@@ -28,7 +28,6 @@ import (
appsv1alpha1 "github.com/apecloud/kubeblocks/apis/apps/v1alpha1"
"github.com/apecloud/kubeblocks/internal/configuration/core"
- "github.com/apecloud/kubeblocks/internal/controller/configuration"
intctrlutil "github.com/apecloud/kubeblocks/internal/controllerutil"
)
@@ -216,15 +215,11 @@ func (r *reconfigureAction) syncReconfigureOperatorStatus(ctx intctrlutil.Reques
}
item := fetcher.ConfigurationObj.Spec.GetConfigurationItem(configSpec.Name)
- status := fetcher.ConfigurationObj.Status.GetItemStatus(configSpec.Name)
- if status == nil || item == nil {
+ if item == nil {
return appsv1alpha1.OpsRunningPhase, nil
}
- if !configuration.IsApplyConfigChanged(fetcher.ConfigMapObj, *item) {
- return appsv1alpha1.OpsRunningPhase, nil
- }
- switch status.Phase {
+ switch intctrlutil.GetConfigSpecReconcilePhase(fetcher.ConfigMapObj, *item, fetcher.ConfigurationObj.Status.GetItemStatus(configSpec.Name)) {
default:
return appsv1alpha1.OpsRunningPhase, nil
case appsv1alpha1.CFailedAndPausePhase:
diff --git a/controllers/apps/operations/reconfigure_test.go b/controllers/apps/operations/reconfigure_test.go
index 5acb8242783..6afad600d5f 100644
--- a/controllers/apps/operations/reconfigure_test.go
+++ b/controllers/apps/operations/reconfigure_test.go
@@ -94,9 +94,7 @@ var _ = Describe("Reconfigure OpsRequest", func() {
var cmObj *corev1.ConfigMap
configuration := builder.NewConfigurationBuilder(testCtx.DefaultNamespace, core.GenerateComponentConfigurationName(clusterName, componentName)).
ClusterRef(clusterName).
- Component(componentName).
- ClusterDefRef(clusterDefinitionName).
- ClusterVerRef(clusterVersionName)
+ Component(componentName)
for _, configSpec := range cdComponent.ConfigSpecs {
cmInsName := core.GetComponentCfgName(clusterName, componentName, configSpec.Name)
cfgCM := testapps.NewCustomizedObj("operations_config/config-template.yaml",
@@ -112,7 +110,7 @@ var _ = Describe("Reconfigure OpsRequest", func() {
constant.CMConfigurationTypeLabelKey, constant.ConfigInstanceType,
),
)
- configuration.AddConfigurationItem(configSpec.Name)
+ configuration.AddConfigurationItem(configSpec)
Expect(testCtx.CheckedCreateObj(ctx, cfgCM)).Should(Succeed())
cmObj = cfgCM
}
@@ -195,7 +193,7 @@ var _ = Describe("Reconfigure OpsRequest", func() {
opsRes, eventContext := assureMockReconfigureData("simple")
reqCtx := intctrlutil.RequestCtx{
Ctx: testCtx.Ctx,
- Log: log.FromContext(ctx).WithValues("Reconfigure"),
+ Log: log.FromContext(ctx).WithName("Reconfigure"),
Recorder: opsRes.Recorder,
}
@@ -260,7 +258,7 @@ var _ = Describe("Reconfigure OpsRequest", func() {
opsRes, eventContext := assureMockReconfigureData("autoReload")
reqCtx := intctrlutil.RequestCtx{
Ctx: testCtx.Ctx,
- Log: log.FromContext(ctx).WithValues("Reconfigure"),
+ Log: log.FromContext(ctx).WithName("Reconfigure"),
Recorder: opsRes.Recorder,
}
diff --git a/controllers/apps/operations/switchover_test.go b/controllers/apps/operations/switchover_test.go
index e81f14c70d0..92f2ef5315a 100644
--- a/controllers/apps/operations/switchover_test.go
+++ b/controllers/apps/operations/switchover_test.go
@@ -153,7 +153,7 @@ var _ = Describe("", func() {
AddContainer(container).
AddAppInstanceLabel(clusterObj.Name).
AddAppComponentLabel(consensusComp).
- AddAppManangedByLabel().
+ AddAppManagedByLabel().
SetReplicas(2).
Create(&testCtx).GetObject()
diff --git a/controllers/apps/operations/switchover_util.go b/controllers/apps/operations/switchover_util.go
index 28b409e6c36..9c75c3a08d3 100644
--- a/controllers/apps/operations/switchover_util.go
+++ b/controllers/apps/operations/switchover_util.go
@@ -35,6 +35,7 @@ import (
appsv1alpha1 "github.com/apecloud/kubeblocks/apis/apps/v1alpha1"
"github.com/apecloud/kubeblocks/controllers/apps/components"
+ "github.com/apecloud/kubeblocks/internal/common"
"github.com/apecloud/kubeblocks/internal/constant"
intctrlcomputil "github.com/apecloud/kubeblocks/internal/controller/component"
intctrlutil "github.com/apecloud/kubeblocks/internal/controllerutil"
@@ -69,8 +70,8 @@ func needDoSwitchover(ctx context.Context,
if err != nil {
return false, err
}
- podParent, _ := components.ParseParentNameAndOrdinal(pod.Name)
- siParent, o := components.ParseParentNameAndOrdinal(switchover.InstanceName)
+ podParent, _ := common.ParseParentNameAndOrdinal(pod.Name)
+ siParent, o := common.ParseParentNameAndOrdinal(switchover.InstanceName)
if podParent != siParent || o < 0 || o >= int32(len(podList.Items)) {
return false, errors.New("switchover.InstanceName is invalid")
}
diff --git a/controllers/apps/operations/switchover_util_test.go b/controllers/apps/operations/switchover_util_test.go
index 5a3a90e83cf..e241cfe06fe 100644
--- a/controllers/apps/operations/switchover_util_test.go
+++ b/controllers/apps/operations/switchover_util_test.go
@@ -98,7 +98,7 @@ var _ = Describe("Switchover Util", func() {
AddContainer(container).
AddAppInstanceLabel(clusterObj.Name).
AddAppComponentLabel(testapps.DefaultRedisCompSpecName).
- AddAppManangedByLabel().
+ AddAppManagedByLabel().
SetReplicas(2).
Create(&testCtx).GetObject()
@@ -153,7 +153,7 @@ var _ = Describe("Switchover Util", func() {
AddContainer(container).
AddAppInstanceLabel(clusterObj.Name).
AddAppComponentLabel(testapps.DefaultRedisCompSpecName).
- AddAppManangedByLabel().
+ AddAppManagedByLabel().
SetReplicas(2).
Create(&testCtx).GetObject()
diff --git a/controllers/apps/operations/util/common_util.go b/controllers/apps/operations/util/common_util.go
index c7667b286fa..cb1047e478f 100644
--- a/controllers/apps/operations/util/common_util.go
+++ b/controllers/apps/operations/util/common_util.go
@@ -27,7 +27,7 @@ import (
"sigs.k8s.io/controller-runtime/pkg/client"
appsv1alpha1 "github.com/apecloud/kubeblocks/apis/apps/v1alpha1"
- dataprotectionv1alpha1 "github.com/apecloud/kubeblocks/apis/dataprotection/v1alpha1"
+ dpv1alpha1 "github.com/apecloud/kubeblocks/apis/dataprotection/v1alpha1"
intctrlutil "github.com/apecloud/kubeblocks/internal/constant"
)
@@ -100,7 +100,7 @@ func GetOpsRequestSliceFromCluster(cluster *appsv1alpha1.Cluster) ([]appsv1alpha
}
// GetOpsRequestFromBackup gets OpsRequest slice from cluster annotations.
-func GetOpsRequestFromBackup(backup *dataprotectionv1alpha1.Backup) *appsv1alpha1.OpsRecorder {
+func GetOpsRequestFromBackup(backup *dpv1alpha1.Backup) *appsv1alpha1.OpsRecorder {
var (
opsRequestName string
opsRequestType string
diff --git a/controllers/apps/operations/util/common_util_test.go b/controllers/apps/operations/util/common_util_test.go
index 70edce24c0e..a2e9eaebd3d 100644
--- a/controllers/apps/operations/util/common_util_test.go
+++ b/controllers/apps/operations/util/common_util_test.go
@@ -26,7 +26,7 @@ import (
"sigs.k8s.io/controller-runtime/pkg/client"
appsv1alpha1 "github.com/apecloud/kubeblocks/apis/apps/v1alpha1"
- dataprotectionv1alpha1 "github.com/apecloud/kubeblocks/apis/dataprotection/v1alpha1"
+ dpv1alpha1 "github.com/apecloud/kubeblocks/apis/dataprotection/v1alpha1"
intctrlutil "github.com/apecloud/kubeblocks/internal/constant"
testapps "github.com/apecloud/kubeblocks/internal/testutil/apps"
)
@@ -97,7 +97,7 @@ var _ = Describe("OpsRequest Controller", func() {
It("Should Test Backup OpsRequest", func() {
By("test GetOpsRequestFromBackup function")
- backup := &dataprotectionv1alpha1.Backup{}
+ backup := &dpv1alpha1.Backup{}
backup.Labels = map[string]string{
intctrlutil.OpsRequestNameLabelKey: "backup-ops",
intctrlutil.OpsRequestTypeLabelKey: string(appsv1alpha1.BackupType),
diff --git a/controllers/apps/opsrequest_controller.go b/controllers/apps/opsrequest_controller.go
index 5340d1c709e..18c3909dfc4 100644
--- a/controllers/apps/opsrequest_controller.go
+++ b/controllers/apps/opsrequest_controller.go
@@ -37,10 +37,9 @@ import (
"sigs.k8s.io/controller-runtime/pkg/handler"
"sigs.k8s.io/controller-runtime/pkg/log"
"sigs.k8s.io/controller-runtime/pkg/reconcile"
- "sigs.k8s.io/controller-runtime/pkg/source"
appsv1alpha1 "github.com/apecloud/kubeblocks/apis/apps/v1alpha1"
- dataprotectionv1alpha1 "github.com/apecloud/kubeblocks/apis/dataprotection/v1alpha1"
+ dpv1alpha1 "github.com/apecloud/kubeblocks/apis/dataprotection/v1alpha1"
"github.com/apecloud/kubeblocks/controllers/apps/operations"
opsutil "github.com/apecloud/kubeblocks/controllers/apps/operations/util"
"github.com/apecloud/kubeblocks/internal/constant"
@@ -84,8 +83,8 @@ func (r *OpsRequestReconciler) Reconcile(ctx context.Context, req ctrl.Request)
func (r *OpsRequestReconciler) SetupWithManager(mgr ctrl.Manager) error {
return ctrl.NewControllerManagedBy(mgr).
For(&appsv1alpha1.OpsRequest{}).
- Watches(&source.Kind{Type: &appsv1alpha1.Cluster{}}, handler.EnqueueRequestsFromMapFunc(r.parseAllOpsRequest)).
- Watches(&source.Kind{Type: &dataprotectionv1alpha1.Backup{}}, handler.EnqueueRequestsFromMapFunc(r.parseBackupOpsRequest)).
+ Watches(&appsv1alpha1.Cluster{}, handler.EnqueueRequestsFromMapFunc(r.parseAllOpsRequest)).
+ Watches(&dpv1alpha1.Backup{}, handler.EnqueueRequestsFromMapFunc(r.parseBackupOpsRequest)).
Complete(r)
}
@@ -285,7 +284,7 @@ func (r *OpsRequestReconciler) handleOpsReqDeletedDuringRunning(reqCtx intctrlut
return nil
}
-func (r *OpsRequestReconciler) parseAllOpsRequest(object client.Object) []reconcile.Request {
+func (r *OpsRequestReconciler) parseAllOpsRequest(ctx context.Context, object client.Object) []reconcile.Request {
cluster := object.(*appsv1alpha1.Cluster)
var (
opsRequestSlice []appsv1alpha1.OpsRecorder
@@ -306,8 +305,8 @@ func (r *OpsRequestReconciler) parseAllOpsRequest(object client.Object) []reconc
return requests
}
-func (r *OpsRequestReconciler) parseBackupOpsRequest(object client.Object) []reconcile.Request {
- backup := object.(*dataprotectionv1alpha1.Backup)
+func (r *OpsRequestReconciler) parseBackupOpsRequest(ctx context.Context, object client.Object) []reconcile.Request {
+ backup := object.(*dpv1alpha1.Backup)
var (
requests []reconcile.Request
)
diff --git a/controllers/apps/opsrequest_controller_test.go b/controllers/apps/opsrequest_controller_test.go
index b588eb823db..6b43a77ea30 100644
--- a/controllers/apps/opsrequest_controller_test.go
+++ b/controllers/apps/opsrequest_controller_test.go
@@ -37,15 +37,15 @@ import (
"sigs.k8s.io/controller-runtime/pkg/client"
appsv1alpha1 "github.com/apecloud/kubeblocks/apis/apps/v1alpha1"
- dataprotectionv1alpha1 "github.com/apecloud/kubeblocks/apis/dataprotection/v1alpha1"
+ dpv1alpha1 "github.com/apecloud/kubeblocks/apis/dataprotection/v1alpha1"
workloads "github.com/apecloud/kubeblocks/apis/workloads/v1alpha1"
opsutil "github.com/apecloud/kubeblocks/controllers/apps/operations/util"
"github.com/apecloud/kubeblocks/internal/constant"
- "github.com/apecloud/kubeblocks/internal/controllerutil"
+ dptypes "github.com/apecloud/kubeblocks/internal/dataprotection/types"
intctrlutil "github.com/apecloud/kubeblocks/internal/generics"
testapps "github.com/apecloud/kubeblocks/internal/testutil/apps"
+ testdp "github.com/apecloud/kubeblocks/internal/testutil/dataprotection"
testk8s "github.com/apecloud/kubeblocks/internal/testutil/k8s"
- viper "github.com/apecloud/kubeblocks/internal/viperx"
lorry "github.com/apecloud/kubeblocks/lorry/client"
)
@@ -181,24 +181,16 @@ var _ = Describe("OpsRequest Controller", func() {
}
var mysqlSts *appsv1.StatefulSet
var mysqlRSM *workloads.ReplicatedStateMachine
- if controllerutil.IsRSMEnabled() {
- rsmList := testk8s.ListAndCheckRSMWithComponent(&testCtx, clusterKey, mysqlCompName)
- mysqlRSM = &rsmList.Items[0]
- mysqlSts = testapps.NewStatefulSetFactory(mysqlRSM.Namespace, mysqlRSM.Name, clusterKey.Name, mysqlCompName).
- SetReplicas(*mysqlRSM.Spec.Replicas).Create(&testCtx).GetObject()
- Expect(testapps.ChangeObjStatus(&testCtx, mysqlSts, func() {
- testk8s.MockStatefulSetReady(mysqlSts)
- })).ShouldNot(HaveOccurred())
- Expect(testapps.ChangeObjStatus(&testCtx, mysqlRSM, func() {
- testk8s.MockRSMReady(mysqlRSM, pod)
- })).ShouldNot(HaveOccurred())
- } else {
- stsList := testk8s.ListAndCheckStatefulSetWithComponent(&testCtx, clusterKey, mysqlCompName)
- mysqlSts = &stsList.Items[0]
- Expect(testapps.ChangeObjStatus(&testCtx, mysqlSts, func() {
- testk8s.MockStatefulSetReady(mysqlSts)
- })).ShouldNot(HaveOccurred())
- }
+ rsmList := testk8s.ListAndCheckRSMWithComponent(&testCtx, clusterKey, mysqlCompName)
+ mysqlRSM = &rsmList.Items[0]
+ mysqlSts = testapps.NewStatefulSetFactory(mysqlRSM.Namespace, mysqlRSM.Name, clusterKey.Name, mysqlCompName).
+ SetReplicas(*mysqlRSM.Spec.Replicas).Create(&testCtx).GetObject()
+ Expect(testapps.ChangeObjStatus(&testCtx, mysqlSts, func() {
+ testk8s.MockStatefulSetReady(mysqlSts)
+ })).ShouldNot(HaveOccurred())
+ Expect(testapps.ChangeObjStatus(&testCtx, mysqlRSM, func() {
+ testk8s.MockRSMReady(mysqlRSM, pod)
+ })).ShouldNot(HaveOccurred())
Eventually(testapps.GetClusterPhase(&testCtx, clusterKey)).Should(Equal(appsv1alpha1.RunningClusterPhase))
By("send VerticalScalingOpsRequest successfully")
@@ -238,15 +230,9 @@ var _ = Describe("OpsRequest Controller", func() {
// })).Should(Succeed())
By("mock bring Cluster and changed component back to running status")
- if controllerutil.IsRSMEnabled() {
- Expect(testapps.GetAndChangeObjStatus(&testCtx, client.ObjectKeyFromObject(mysqlRSM), func(tmpRSM *workloads.ReplicatedStateMachine) {
- testk8s.MockRSMReady(tmpRSM, pod)
- })()).ShouldNot(HaveOccurred())
- } else {
- Expect(testapps.GetAndChangeObjStatus(&testCtx, client.ObjectKeyFromObject(mysqlSts), func(tmpSts *appsv1.StatefulSet) {
- testk8s.MockStatefulSetReady(tmpSts)
- })()).ShouldNot(HaveOccurred())
- }
+ Expect(testapps.GetAndChangeObjStatus(&testCtx, client.ObjectKeyFromObject(mysqlRSM), func(tmpRSM *workloads.ReplicatedStateMachine) {
+ testk8s.MockRSMReady(tmpRSM, pod)
+ })()).ShouldNot(HaveOccurred())
Eventually(testapps.GetClusterComponentPhase(&testCtx, clusterKey, mysqlCompName)).Should(Equal(appsv1alpha1.RunningClusterCompPhase))
Eventually(testapps.GetClusterPhase(&testCtx, clusterKey)).Should(Equal(appsv1alpha1.RunningClusterPhase))
// checkLatestOpsHasProcessed(clusterKey)
@@ -273,15 +259,9 @@ var _ = Describe("OpsRequest Controller", func() {
targetRequests = scalingCtx.target.resource.Requests
}
- if controllerutil.IsRSMEnabled() {
- rsmList := testk8s.ListAndCheckRSMWithComponent(&testCtx, clusterKey, mysqlCompName)
- mysqlRSM = &rsmList.Items[0]
- Expect(reflect.DeepEqual(mysqlRSM.Spec.Template.Spec.Containers[0].Resources.Requests, targetRequests)).Should(BeTrue())
- } else {
- stsList := testk8s.ListAndCheckStatefulSetWithComponent(&testCtx, clusterKey, mysqlCompName)
- mysqlSts = &stsList.Items[0]
- Expect(reflect.DeepEqual(mysqlSts.Spec.Template.Spec.Containers[0].Resources.Requests, targetRequests)).Should(BeTrue())
- }
+ rsmList = testk8s.ListAndCheckRSMWithComponent(&testCtx, clusterKey, mysqlCompName)
+ mysqlRSM = &rsmList.Items[0]
+ Expect(reflect.DeepEqual(mysqlRSM.Spec.Template.Spec.Containers[0].Resources.Requests, targetRequests)).Should(BeTrue())
By("check OpsRequest reclaimed after ttl")
Expect(testapps.ChangeObj(&testCtx, verticalScalingOpsRequest, func(lopsReq *appsv1alpha1.OpsRequest) {
@@ -343,7 +323,7 @@ var _ = Describe("OpsRequest Controller", func() {
Context("with Cluster which has MySQL ConsensusSet", func() {
BeforeEach(func() {
By("Create a clusterDefinition obj")
- viper.Set("VOLUMESNAPSHOT", true)
+ testk8s.MockEnableVolumeSnapshot(&testCtx, testk8s.DefaultStorageClassName)
clusterDefObj = testapps.NewClusterDefFactory(clusterDefName).
AddComponentDef(testapps.ConsensusMySQLComponent, mysqlCompDefName).
AddHorizontalScalePolicy(appsv1alpha1.HorizontalScalePolicy{
@@ -358,32 +338,22 @@ var _ = Describe("OpsRequest Controller", func() {
})
componentWorkload := func() client.Object {
- if controllerutil.IsRSMEnabled() {
- rsmList := testk8s.ListAndCheckRSMWithComponent(&testCtx, clusterKey, mysqlCompName)
- return &rsmList.Items[0]
- }
- stsList := testk8s.ListAndCheckStatefulSetWithComponent(&testCtx, clusterKey, mysqlCompName)
- return &stsList.Items[0]
+ rsmList := testk8s.ListAndCheckRSMWithComponent(&testCtx, clusterKey, mysqlCompName)
+ return &rsmList.Items[0]
}
mockCompRunning := func(replicas int32) {
- var sts *appsv1.StatefulSet
wl := componentWorkload()
- if controllerutil.IsRSMEnabled() {
- rsm, _ := wl.(*workloads.ReplicatedStateMachine)
- sts = testapps.NewStatefulSetFactory(rsm.Namespace, rsm.Name, clusterKey.Name, mysqlCompName).
- SetReplicas(*rsm.Spec.Replicas).GetObject()
- testapps.CheckedCreateK8sResource(&testCtx, sts)
- } else {
- sts, _ = wl.(*appsv1.StatefulSet)
- }
+ rsm, _ := wl.(*workloads.ReplicatedStateMachine)
+ sts := testapps.NewStatefulSetFactory(rsm.Namespace, rsm.Name, clusterKey.Name, mysqlCompName).
+ SetReplicas(*rsm.Spec.Replicas).GetObject()
+ testapps.CheckedCreateK8sResource(&testCtx, sts)
+
mockPods := testapps.MockConsensusComponentPods(&testCtx, sts, clusterObj.Name, mysqlCompName)
- if controllerutil.IsRSMEnabled() {
- rsm, _ := wl.(*workloads.ReplicatedStateMachine)
- Expect(testapps.ChangeObjStatus(&testCtx, rsm, func() {
- testk8s.MockRSMReady(rsm, mockPods...)
- })).ShouldNot(HaveOccurred())
- }
+ rsm, _ = wl.(*workloads.ReplicatedStateMachine)
+ Expect(testapps.ChangeObjStatus(&testCtx, rsm, func() {
+ testk8s.MockRSMReady(rsm, mockPods...)
+ })).ShouldNot(HaveOccurred())
Expect(testapps.ChangeObjStatus(&testCtx, sts, func() {
testk8s.MockStatefulSetReady(sts)
})).ShouldNot(HaveOccurred())
@@ -395,7 +365,7 @@ var _ = Describe("OpsRequest Controller", func() {
createBackupPolicyTpl(clusterDefObj)
By("set component to horizontal with snapshot policy and create a cluster")
- viper.Set("VOLUMESNAPSHOT", true)
+ testk8s.MockEnableVolumeSnapshot(&testCtx, testk8s.DefaultStorageClassName)
if clusterDefObj.Spec.ComponentDefs[0].HorizontalScalePolicy == nil {
Expect(testapps.GetAndChangeObj(&testCtx, client.ObjectKeyFromObject(clusterDefObj),
func(clusterDef *appsv1alpha1.ClusterDefinition) {
@@ -419,7 +389,11 @@ var _ = Describe("OpsRequest Controller", func() {
for i := 0; i < int(replicas); i++ {
pvcName := fmt.Sprintf("%s-%s-%s-%d", testapps.DataVolumeName, clusterKey.Name, mysqlCompName, i)
pvc := testapps.NewPersistentVolumeClaimFactory(testCtx.DefaultNamespace, pvcName, clusterKey.Name,
- mysqlCompName, testapps.DataVolumeName).SetStorage("1Gi").Create(&testCtx).GetObject()
+ mysqlCompName, testapps.DataVolumeName).
+ SetStorage("1Gi").
+ SetStorageClass(testk8s.DefaultStorageClassName).
+ Create(&testCtx).
+ GetObject()
// mock pvc bound
Expect(testapps.GetAndChangeObjStatus(&testCtx, client.ObjectKeyFromObject(pvc), func(pvc *corev1.PersistentVolumeClaim) {
pvc.Status.Phase = corev1.ClaimBound
@@ -460,7 +434,7 @@ var _ = Describe("OpsRequest Controller", func() {
It("HorizontalScaling when not support snapshot", func() {
By("init backup policy template, mysql cluster and hscale ops")
- viper.Set("VOLUMESNAPSHOT", false)
+ testk8s.MockDisableVolumeSnapshot(&testCtx, testk8s.DefaultStorageClassName)
createMysqlCluster(3)
cluster := &appsv1alpha1.Cluster{}
@@ -502,8 +476,9 @@ var _ = Describe("OpsRequest Controller", func() {
It("HorizontalScaling via volume snapshot backup", func() {
By("init backup policy template, mysql cluster and hscale ops")
- viper.Set("VOLUMESNAPSHOT", true)
- createMysqlCluster(3)
+ testk8s.MockEnableVolumeSnapshot(&testCtx, testk8s.DefaultStorageClassName)
+ oldReplicas := int32(3)
+ createMysqlCluster(oldReplicas)
replicas := int32(5)
ops := createClusterHscaleOps(replicas)
@@ -520,11 +495,12 @@ var _ = Describe("OpsRequest Controller", func() {
})).Should(Succeed())
By("mock backup status is ready, component phase should change to Updating when component is horizontally scaling.")
- backupKey := types.NamespacedName{Name: fmt.Sprintf("%s-%s-scaling",
+ backupKey := client.ObjectKey{Name: fmt.Sprintf("%s-%s-scaling",
clusterKey.Name, mysqlCompName), Namespace: testCtx.DefaultNamespace}
- backup := &dataprotectionv1alpha1.Backup{}
+ backup := &dpv1alpha1.Backup{}
Expect(k8sClient.Get(testCtx.Ctx, backupKey, backup)).Should(Succeed())
- backup.Status.Phase = dataprotectionv1alpha1.BackupCompleted
+ backup.Status.Phase = dpv1alpha1.BackupPhaseCompleted
+ testdp.MockBackupStatusMethod(backup, testapps.DataVolumeName)
Expect(k8sClient.Status().Update(testCtx.Ctx, backup)).Should(Succeed())
Eventually(testapps.CheckObj(&testCtx, clusterKey, func(g Gomega, cluster *appsv1alpha1.Cluster) {
g.Expect(cluster.Status.Components[mysqlCompName].Phase).Should(Equal(appsv1alpha1.UpdatingClusterCompPhase))
@@ -536,7 +512,7 @@ var _ = Describe("OpsRequest Controller", func() {
vs.Name = backupKey.Name
vs.Namespace = backupKey.Namespace
vs.Labels = map[string]string{
- constant.DataProtectionLabelBackupNameKey: backupKey.Name,
+ dptypes.DataProtectionLabelBackupNameKey: backupKey.Name,
}
pvcName := ""
vs.Spec = snapshotv1.VolumeSnapshotSpec{
@@ -547,17 +523,44 @@ var _ = Describe("OpsRequest Controller", func() {
Expect(k8sClient.Create(testCtx.Ctx, vs)).Should(Succeed())
Eventually(testapps.CheckObjExists(&testCtx, backupKey, vs, true)).Should(Succeed())
+ mockComponentPVCsAndBound := func(comp *appsv1alpha1.ClusterComponentSpec) {
+ for i := 0; i < int(replicas); i++ {
+ for _, vct := range comp.VolumeClaimTemplates {
+ pvcKey := types.NamespacedName{
+ Namespace: clusterKey.Namespace,
+ Name: fmt.Sprintf("%s-%s-%s-%d", vct.Name, clusterKey.Name, comp.Name, i),
+ }
+ testapps.NewPersistentVolumeClaimFactory(testCtx.DefaultNamespace, pvcKey.Name, clusterKey.Name,
+ comp.Name, testapps.DataVolumeName).SetStorage(vct.Spec.Resources.Requests.Storage().String()).AddLabelsInMap(map[string]string{
+ constant.AppInstanceLabelKey: clusterKey.Name,
+ constant.KBAppComponentLabelKey: comp.Name,
+ constant.AppManagedByLabelKey: constant.AppName,
+ }).CheckedCreate(&testCtx)
+ Eventually(testapps.GetAndChangeObjStatus(&testCtx, pvcKey, func(pvc *corev1.PersistentVolumeClaim) {
+ pvc.Status.Phase = corev1.ClaimBound
+ if pvc.Status.Capacity == nil {
+ pvc.Status.Capacity = corev1.ResourceList{}
+ }
+ pvc.Status.Capacity[corev1.ResourceStorage] = pvc.Spec.Resources.Requests[corev1.ResourceStorage]
+ })).Should(Succeed())
+ }
+ }
+ }
+
+ // mock pvcs have restored
+ mockComponentPVCsAndBound(clusterObj.Spec.GetComponentByName(mysqlCompName))
+ // check restore CR and mock it to Completed
+ checkRestoreAndSetCompleted(clusterKey, mysqlCompName, int(replicas-oldReplicas))
+
By("check the underlying workload been updated")
- if controllerutil.IsRSMEnabled() {
- Eventually(testapps.CheckObj(&testCtx, client.ObjectKeyFromObject(componentWorkload()),
- func(g Gomega, rsm *workloads.ReplicatedStateMachine) {
- g.Expect(*rsm.Spec.Replicas).Should(Equal(replicas))
- })).Should(Succeed())
- rsm := componentWorkload()
- Eventually(testapps.GetAndChangeObj(&testCtx, client.ObjectKeyFromObject(rsm), func(sts *appsv1.StatefulSet) {
- sts.Spec.Replicas = &replicas
+ Eventually(testapps.CheckObj(&testCtx, client.ObjectKeyFromObject(componentWorkload()),
+ func(g Gomega, rsm *workloads.ReplicatedStateMachine) {
+ g.Expect(*rsm.Spec.Replicas).Should(Equal(replicas))
})).Should(Succeed())
- }
+ rsm := componentWorkload()
+ Eventually(testapps.GetAndChangeObj(&testCtx, client.ObjectKeyFromObject(rsm), func(sts *appsv1.StatefulSet) {
+ sts.Spec.Replicas = &replicas
+ })).Should(Succeed())
Eventually(testapps.CheckObj(&testCtx, client.ObjectKeyFromObject(componentWorkload()),
func(g Gomega, sts *appsv1.StatefulSet) {
g.Expect(*sts.Spec.Replicas).Should(Equal(replicas))
@@ -629,16 +632,14 @@ var _ = Describe("OpsRequest Controller", func() {
})).Should(Succeed())
By("check the underlying workload been updated")
- if controllerutil.IsRSMEnabled() {
- Eventually(testapps.CheckObj(&testCtx, client.ObjectKeyFromObject(componentWorkload()),
- func(g Gomega, rsm *workloads.ReplicatedStateMachine) {
- g.Expect(*rsm.Spec.Replicas).Should(Equal(replicas))
- })).Should(Succeed())
- rsm := componentWorkload()
- Eventually(testapps.GetAndChangeObj(&testCtx, client.ObjectKeyFromObject(rsm), func(sts *appsv1.StatefulSet) {
- sts.Spec.Replicas = &replicas
+ Eventually(testapps.CheckObj(&testCtx, client.ObjectKeyFromObject(componentWorkload()),
+ func(g Gomega, rsm *workloads.ReplicatedStateMachine) {
+ g.Expect(*rsm.Spec.Replicas).Should(Equal(replicas))
})).Should(Succeed())
- }
+ rsm := componentWorkload()
+ Eventually(testapps.GetAndChangeObj(&testCtx, client.ObjectKeyFromObject(rsm), func(sts *appsv1.StatefulSet) {
+ sts.Spec.Replicas = &replicas
+ })).Should(Succeed())
Eventually(testapps.CheckObj(&testCtx, client.ObjectKeyFromObject(componentWorkload()),
func(g Gomega, sts *appsv1.StatefulSet) {
g.Expect(*sts.Spec.Replicas).Should(Equal(replicas))
@@ -659,7 +660,7 @@ var _ = Describe("OpsRequest Controller", func() {
It("delete Running opsRequest", func() {
By("Create a horizontalScaling ops")
- viper.Set("VOLUMESNAPSHOT", true)
+ testk8s.MockEnableVolumeSnapshot(&testCtx, testk8s.DefaultStorageClassName)
createMysqlCluster(3)
ops := createClusterHscaleOps(5)
opsKey := client.ObjectKeyFromObject(ops)
@@ -687,7 +688,7 @@ var _ = Describe("OpsRequest Controller", func() {
It("cancel HorizontalScaling opsRequest which is Running", func() {
By("create cluster and mock it to running")
- viper.Set("VOLUMESNAPSHOT", false)
+ testk8s.MockDisableVolumeSnapshot(&testCtx, testk8s.DefaultStorageClassName)
oldReplicas := int32(3)
createMysqlCluster(oldReplicas)
mockCompRunning(oldReplicas)
diff --git a/controllers/apps/suite_test.go b/controllers/apps/suite_test.go
index e668891f266..64aabda4349 100644
--- a/controllers/apps/suite_test.go
+++ b/controllers/apps/suite_test.go
@@ -42,7 +42,7 @@ import (
"sigs.k8s.io/controller-runtime/pkg/log/zap"
appsv1alpha1 "github.com/apecloud/kubeblocks/apis/apps/v1alpha1"
- dataprotectionv1alpha1 "github.com/apecloud/kubeblocks/apis/dataprotection/v1alpha1"
+ dpv1alpha1 "github.com/apecloud/kubeblocks/apis/dataprotection/v1alpha1"
workloads "github.com/apecloud/kubeblocks/apis/workloads/v1alpha1"
"github.com/apecloud/kubeblocks/controllers/apps/configuration"
"github.com/apecloud/kubeblocks/controllers/k8score"
@@ -116,7 +116,7 @@ var _ = BeforeSuite(func() {
err = appsv1alpha1.AddToScheme(scheme.Scheme)
Expect(err).NotTo(HaveOccurred())
- err = dataprotectionv1alpha1.AddToScheme(scheme.Scheme)
+ err = dpv1alpha1.AddToScheme(scheme.Scheme)
Expect(err).NotTo(HaveOccurred())
err = snapshotv1.AddToScheme(scheme.Scheme)
diff --git a/controllers/apps/systemaccount_controller.go b/controllers/apps/systemaccount_controller.go
index 2a4b47e0833..0b32e36077b 100644
--- a/controllers/apps/systemaccount_controller.go
+++ b/controllers/apps/systemaccount_controller.go
@@ -38,7 +38,6 @@ import (
"sigs.k8s.io/controller-runtime/pkg/event"
"sigs.k8s.io/controller-runtime/pkg/handler"
"sigs.k8s.io/controller-runtime/pkg/log"
- "sigs.k8s.io/controller-runtime/pkg/source"
appsv1alpha1 "github.com/apecloud/kubeblocks/apis/apps/v1alpha1"
opsutil "github.com/apecloud/kubeblocks/controllers/apps/operations/util"
@@ -292,7 +291,7 @@ func (r *SystemAccountReconciler) SetupWithManager(mgr ctrl.Manager) error {
return ctrl.NewControllerManagedBy(mgr).
For(&appsv1alpha1.Cluster{}).
Owns(&corev1.Secret{}).
- Watches(&source.Kind{Type: &batchv1.Job{}}, r.jobCompletionHandler()).
+ Watches(&batchv1.Job{}, r.jobCompletionHandler()).
Complete(r)
}
@@ -469,7 +468,7 @@ func (r *SystemAccountReconciler) jobCompletionHandler() *handler.Funcs {
// 2. has completed (either successed or failed)
// 3. is under deletion (either by user or by TTL, where deletionTimestamp is set)
return &handler.Funcs{
- UpdateFunc: func(e event.UpdateEvent, q workqueue.RateLimitingInterface) {
+ UpdateFunc: func(ctx context.Context, e event.UpdateEvent, q workqueue.RateLimitingInterface) {
var (
jobTerminated = false
job *batchv1.Job
@@ -554,6 +553,6 @@ func (r *SystemAccountReconciler) jobCompletionHandler() *handler.Funcs {
// existsOperations checks if the cluster is doing operations
func existsOperations(cluster *appsv1alpha1.Cluster) bool {
opsRequestMap, _ := opsutil.GetOpsRequestSliceFromCluster(cluster)
- _, isRestoring := cluster.Annotations[constant.RestoreFromBackUpAnnotationKey]
+ _, isRestoring := cluster.Annotations[constant.RestoreFromBackupAnnotationKey]
return len(opsRequestMap) > 0 || isRestoring
}
diff --git a/controllers/apps/tls_utils_test.go b/controllers/apps/tls_utils_test.go
index 4ac61fed483..beb56273bd5 100644
--- a/controllers/apps/tls_utils_test.go
+++ b/controllers/apps/tls_utils_test.go
@@ -26,7 +26,6 @@ import (
. "github.com/onsi/ginkgo/v2"
. "github.com/onsi/gomega"
- appsv1 "k8s.io/api/apps/v1"
corev1 "k8s.io/api/core/v1"
apierrors "k8s.io/apimachinery/pkg/api/errors"
"k8s.io/apimachinery/pkg/types"
@@ -37,7 +36,6 @@ import (
cfgcore "github.com/apecloud/kubeblocks/internal/configuration/core"
"github.com/apecloud/kubeblocks/internal/constant"
"github.com/apecloud/kubeblocks/internal/controller/plan"
- "github.com/apecloud/kubeblocks/internal/controllerutil"
"github.com/apecloud/kubeblocks/internal/generics"
testapps "github.com/apecloud/kubeblocks/internal/testutil/apps"
testk8s "github.com/apecloud/kubeblocks/internal/testutil/k8s"
@@ -298,14 +296,9 @@ var _ = Describe("TLS self-signed cert function", func() {
Eventually(k8sClient.Get(ctx, clusterKey, clusterObj)).Should(Succeed())
Eventually(testapps.GetClusterObservedGeneration(&testCtx, clusterKey)).Should(BeEquivalentTo(1))
Eventually(testapps.GetClusterPhase(&testCtx, clusterKey)).Should(Equal(appsv1alpha1.CreatingClusterPhase))
- var sts appsv1.StatefulSet
- if controllerutil.IsRSMEnabled() {
- rsmList := testk8s.ListAndCheckRSM(&testCtx, clusterKey)
- sts = *components.ConvertRSMToSTS(&rsmList.Items[0])
- } else {
- stsList := testk8s.ListAndCheckStatefulSet(&testCtx, clusterKey)
- sts = stsList.Items[0]
- }
+
+ rsmList := testk8s.ListAndCheckRSM(&testCtx, clusterKey)
+ sts := *components.ConvertRSMToSTS(&rsmList.Items[0])
cd := &appsv1alpha1.ClusterDefinition{}
Expect(k8sClient.Get(ctx, types.NamespacedName{Name: clusterDefName, Namespace: testCtx.DefaultNamespace}, cd)).Should(Succeed())
cmName := cfgcore.GetInstanceCMName(&sts, &cd.Spec.ComponentDefs[0].ConfigSpecs[0].ComponentTemplateSpec)
diff --git a/controllers/apps/transform_restore.go b/controllers/apps/transform_restore.go
index 40cfdb61493..fc3ecabe729 100644
--- a/controllers/apps/transform_restore.go
+++ b/controllers/apps/transform_restore.go
@@ -54,19 +54,18 @@ func (t *RestoreTransformer) Transform(ctx graph.TransformContext, dag *graph.DA
return err
}
for _, spec := range cluster.Spec.ComponentSpecs {
+ if cluster.Annotations[constant.RestoreFromBackupAnnotationKey] == "" {
+ continue
+ }
+
comp, err := components.NewComponent(reqCtx, t.Client, clusterDef, clusterVer, cluster, spec.Name, nil)
if err != nil {
return err
}
syncComp := comp.GetSynthesizedComponent()
- if cluster.Annotations[constant.RestoreFromBackUpAnnotationKey] != "" {
- if err = plan.DoRestore(reqCtx.Ctx, t.Client, cluster, syncComp, rscheme); err != nil {
- return commitError(err)
- }
- } else if cluster.Annotations[constant.RestoreFromTimeAnnotationKey] != "" {
- if err = plan.DoPITR(reqCtx.Ctx, t.Client, cluster, syncComp, rscheme); err != nil {
- return commitError(err)
- }
+ restoreMGR := plan.NewRestoreManager(reqCtx.Ctx, t.Client, cluster, rscheme, nil, syncComp.Replicas, 0)
+ if err = restoreMGR.DoRestore(syncComp); err != nil {
+ return commitError(err)
}
}
return nil
diff --git a/controllers/apps/transform_types.go b/controllers/apps/transform_types.go
index 655013f6c3c..6cbfe141829 100644
--- a/controllers/apps/transform_types.go
+++ b/controllers/apps/transform_types.go
@@ -29,7 +29,7 @@ import (
"sigs.k8s.io/controller-runtime/pkg/client"
appsv1alpha1 "github.com/apecloud/kubeblocks/apis/apps/v1alpha1"
- dataprotectionv1alpha1 "github.com/apecloud/kubeblocks/apis/dataprotection/v1alpha1"
+ dpv1alpha1 "github.com/apecloud/kubeblocks/apis/dataprotection/v1alpha1"
extensionsv1alpha1 "github.com/apecloud/kubeblocks/apis/extensions/v1alpha1"
workloads "github.com/apecloud/kubeblocks/apis/workloads/v1alpha1"
client2 "github.com/apecloud/kubeblocks/internal/controller/client"
@@ -43,7 +43,7 @@ func init() {
utilruntime.Must(clientgoscheme.AddToScheme(rscheme))
utilruntime.Must(appsv1alpha1.AddToScheme(rscheme))
- utilruntime.Must(dataprotectionv1alpha1.AddToScheme(rscheme))
+ utilruntime.Must(dpv1alpha1.AddToScheme(rscheme))
utilruntime.Must(snapshotv1.AddToScheme(rscheme))
utilruntime.Must(extensionsv1alpha1.AddToScheme(rscheme))
utilruntime.Must(batchv1.AddToScheme(rscheme))
diff --git a/controllers/apps/transformer_backup_policy_tpl.go b/controllers/apps/transformer_backup_policy_tpl.go
index bbac92002e3..c52382461ff 100644
--- a/controllers/apps/transformer_backup_policy_tpl.go
+++ b/controllers/apps/transformer_backup_policy_tpl.go
@@ -24,213 +24,360 @@ import (
"golang.org/x/exp/slices"
corev1 "k8s.io/api/core/v1"
- apierrors "k8s.io/apimachinery/pkg/api/errors"
- "k8s.io/apimachinery/pkg/api/resource"
metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
"sigs.k8s.io/controller-runtime/pkg/client"
appsv1alpha1 "github.com/apecloud/kubeblocks/apis/apps/v1alpha1"
- dataprotectionv1alpha1 "github.com/apecloud/kubeblocks/apis/dataprotection/v1alpha1"
+ dpv1alpha1 "github.com/apecloud/kubeblocks/apis/dataprotection/v1alpha1"
+ workloads "github.com/apecloud/kubeblocks/apis/workloads/v1alpha1"
"github.com/apecloud/kubeblocks/internal/constant"
"github.com/apecloud/kubeblocks/internal/controller/graph"
ictrltypes "github.com/apecloud/kubeblocks/internal/controller/types"
intctrlutil "github.com/apecloud/kubeblocks/internal/controllerutil"
- viper "github.com/apecloud/kubeblocks/internal/viperx"
+ dptypes "github.com/apecloud/kubeblocks/internal/dataprotection/types"
)
-// BackupPolicyTPLTransformer transforms the backup policy template to the backup policy.
-type BackupPolicyTPLTransformer struct {
+// BackupPolicyTplTransformer transforms the backup policy template to the backup policy.
+type BackupPolicyTplTransformer struct {
+ *ClusterTransformContext
+
tplCount int
tplIdentifier string
isDefaultTemplate string
+
+ backupPolicyTpl *appsv1alpha1.BackupPolicyTemplate
+ backupPolicy *appsv1alpha1.BackupPolicy
+ compWorkloadType appsv1alpha1.WorkloadType
}
-var _ graph.Transformer = &BackupPolicyTPLTransformer{}
+var _ graph.Transformer = &BackupPolicyTplTransformer{}
+
+// Transform transforms the backup policy template to the backup policy and
+// backup schedule.
+func (r *BackupPolicyTplTransformer) Transform(ctx graph.TransformContext, dag *graph.DAG) error {
+ rootVertex, err := ictrltypes.FindRootVertex(dag)
+ if err != nil {
+ return err
+ }
-func (r *BackupPolicyTPLTransformer) Transform(ctx graph.TransformContext, dag *graph.DAG) error {
- transCtx, _ := ctx.(*ClusterTransformContext)
- clusterDefName := transCtx.ClusterDef.Name
- backupPolicyTPLs := &appsv1alpha1.BackupPolicyTemplateList{}
- if err := transCtx.Client.List(transCtx.Context, backupPolicyTPLs, client.MatchingLabels{constant.ClusterDefLabelKey: clusterDefName}); err != nil {
+ r.ClusterTransformContext = ctx.(*ClusterTransformContext)
+ clusterDefName := r.ClusterDef.Name
+ backupPolicyTpls := &appsv1alpha1.BackupPolicyTemplateList{}
+ if err = r.Client.List(r.Context, backupPolicyTpls,
+ client.MatchingLabels{constant.ClusterDefLabelKey: clusterDefName}); err != nil {
return err
}
- r.tplCount = len(backupPolicyTPLs.Items)
+ r.tplCount = len(backupPolicyTpls.Items)
if r.tplCount == 0 {
return nil
}
- rootVertex, err := ictrltypes.FindRootVertex(dag)
- if err != nil {
- return err
- }
- origCluster := transCtx.OrigCluster
+
backupPolicyNames := map[string]struct{}{}
- for _, tpl := range backupPolicyTPLs.Items {
- r.isDefaultTemplate = tpl.Annotations[constant.DefaultBackupPolicyTemplateAnnotationKey]
+ backupScheduleNames := map[string]struct{}{}
+ for _, tpl := range backupPolicyTpls.Items {
+ r.isDefaultTemplate = tpl.Annotations[dptypes.DefaultBackupPolicyTemplateAnnotationKey]
r.tplIdentifier = tpl.Spec.Identifier
- for _, v := range tpl.Spec.BackupPolicies {
- compDef := transCtx.ClusterDef.GetComponentDefByName(v.ComponentDefRef)
+ r.backupPolicyTpl = &tpl
+
+ for i, bp := range tpl.Spec.BackupPolicies {
+ compDef := r.ClusterDef.GetComponentDefByName(bp.ComponentDefRef)
if compDef == nil {
- return intctrlutil.NewNotFound("componentDef %s not found in ClusterDefinition: %s ", v.ComponentDefRef, clusterDefName)
+ return intctrlutil.NewNotFound("componentDef %s not found in ClusterDefinition: %s ",
+ bp.ComponentDefRef, clusterDefName)
}
- // build the backup policy from the template.
- backupPolicy, action := r.transformBackupPolicy(transCtx, v, origCluster, compDef.WorkloadType, &tpl)
-
- // merge cluster backup configuration into the backup policy.
- r.mergeClusterBackup(transCtx, origCluster, backupPolicy)
- if backupPolicy == nil {
- continue
+ r.backupPolicy = &tpl.Spec.BackupPolicies[i]
+ r.compWorkloadType = compDef.WorkloadType
+
+ transformBackupPolicy := func() (*dpv1alpha1.BackupPolicy, *ictrltypes.LifecycleVertex) {
+ // build the data protection backup policy from the template.
+ dpBackupPolicy, action := r.transformBackupPolicy()
+ if dpBackupPolicy == nil {
+ return nil, nil
+ }
+
+ // if exist multiple backup policy templates and duplicate spec.identifier,
+ // the generated backupPolicy may have duplicate names, so it is
+ // necessary to check if it already exists.
+ if _, ok := backupPolicyNames[dpBackupPolicy.Name]; ok {
+ return dpBackupPolicy, nil
+ }
+
+ vertex := &ictrltypes.LifecycleVertex{Obj: dpBackupPolicy, Action: action}
+ dag.AddVertex(vertex)
+ dag.Connect(rootVertex, vertex)
+ backupPolicyNames[dpBackupPolicy.Name] = struct{}{}
+ return dpBackupPolicy, vertex
}
- // if exist multiple backup policy templates and duplicate spec.identifier,
- // the backupPolicy that may be generated may have duplicate names, and it is necessary to check if it already exists.
- if _, ok := backupPolicyNames[backupPolicy.Name]; ok {
- continue
+ transformBackupSchedule := func(
+ backupPolicy *dpv1alpha1.BackupPolicy,
+ bpVertex *ictrltypes.LifecycleVertex) {
+ // if backup policy is nil, it means that the backup policy template
+ // is invalid, backup schedule depends on backup policy, so we do
+ // not need to transform backup schedule.
+ if backupPolicy == nil {
+ return
+ }
+
+ // only create backup schedule for the default backup policy template
+ // if there are multiple backup policy templates.
+ if r.isDefaultTemplate != trueVal && r.tplCount > 1 {
+ return
+ }
+
+ // build the data protection backup schedule from the template.
+ dpBackupSchedule, action := r.transformBackupSchedule(backupPolicy)
+
+ // merge cluster backup configuration into the backup schedule.
+ // If the backup schedule is nil, create a new backup schedule
+ // based on the cluster backup configuration.
+ if dpBackupSchedule == nil {
+ action = ictrltypes.ActionCreatePtr()
+ } else if action == nil {
+ action = ictrltypes.ActionUpdatePtr()
+ }
+
+ // for a cluster, the default backup schedule is created by backup
+ // policy template, user can also configure cluster backup in the
+ // cluster custom object, such as enable cluster backup, set backup
+ // schedule, etc.
+ // We always prioritize the cluster backup configuration in the
+ // cluster object, so we need to merge the cluster backup configuration
+ // into the default backup schedule created by backup policy template
+ // if it exists.
+ dpBackupSchedule = r.mergeClusterBackup(backupPolicy, dpBackupSchedule)
+ if dpBackupSchedule == nil {
+ return
+ }
+
+ // if exist multiple backup policy templates and duplicate spec.identifier,
+ // the backupPolicy that may be generated may have duplicate names,
+ // and it is necessary to check if it already exists.
+ if _, ok := backupScheduleNames[dpBackupSchedule.Name]; ok {
+ return
+ }
+
+ parent := rootVertex
+ if bpVertex != nil {
+ parent = bpVertex
+ }
+ vertex := &ictrltypes.LifecycleVertex{Obj: dpBackupSchedule, Action: action}
+ dag.AddVertex(vertex)
+ dag.Connect(parent, vertex)
+ backupScheduleNames[dpBackupSchedule.Name] = struct{}{}
}
- vertex := &ictrltypes.LifecycleVertex{Obj: backupPolicy, Action: action}
- dag.AddVertex(vertex)
- dag.Connect(rootVertex, vertex)
- backupPolicyNames[backupPolicy.Name] = struct{}{}
+
+ // transform backup policy template to data protection backupPolicy
+ // and backupSchedule
+ policy, policyVertex := transformBackupPolicy()
+ transformBackupSchedule(policy, policyVertex)
}
}
return nil
}
// transformBackupPolicy transforms backup policy template to backup policy.
-func (r *BackupPolicyTPLTransformer) transformBackupPolicy(transCtx *ClusterTransformContext,
- policyTPL appsv1alpha1.BackupPolicy,
- cluster *appsv1alpha1.Cluster,
- workloadType appsv1alpha1.WorkloadType,
- tpl *appsv1alpha1.BackupPolicyTemplate) (*dataprotectionv1alpha1.BackupPolicy, *ictrltypes.LifecycleAction) {
- backupPolicyName := DeriveBackupPolicyName(cluster.Name, policyTPL.ComponentDefRef, r.tplIdentifier)
- backupPolicy := &dataprotectionv1alpha1.BackupPolicy{}
- if err := transCtx.Client.Get(transCtx.Context, client.ObjectKey{Namespace: cluster.Namespace, Name: backupPolicyName}, backupPolicy); err != nil && !apierrors.IsNotFound(err) {
+func (r *BackupPolicyTplTransformer) transformBackupPolicy() (*dpv1alpha1.BackupPolicy, *ictrltypes.LifecycleAction) {
+ cluster := r.OrigCluster
+ backupPolicyName := generateBackupPolicyName(cluster.Name, r.backupPolicy.ComponentDefRef, r.tplIdentifier)
+ backupPolicy := &dpv1alpha1.BackupPolicy{}
+ if err := r.Client.Get(r.Context, client.ObjectKey{
+ Namespace: cluster.Namespace,
+ Name: backupPolicyName,
+ }, backupPolicy); client.IgnoreNotFound(err) != nil {
return nil, nil
}
+
if len(backupPolicy.Name) == 0 {
- // build a new backup policy from the backup policy template.
- return r.buildBackupPolicy(policyTPL, cluster, workloadType, tpl, backupPolicyName), ictrltypes.ActionCreatePtr()
+ // build a new backup policy by the backup policy template.
+ return r.buildBackupPolicy(backupPolicyName), ictrltypes.ActionCreatePtr()
}
+
// sync the existing backup policy with the cluster changes
- r.syncBackupPolicy(backupPolicy, cluster, policyTPL, workloadType, tpl)
+ r.syncBackupPolicy(backupPolicy)
return backupPolicy, ictrltypes.ActionUpdatePtr()
}
+func (r *BackupPolicyTplTransformer) transformBackupSchedule(
+ backupPolicy *dpv1alpha1.BackupPolicy) (*dpv1alpha1.BackupSchedule, *ictrltypes.LifecycleAction) {
+ cluster := r.OrigCluster
+ scheduleName := generateBackupScheduleName(cluster.Name, r.backupPolicy.ComponentDefRef, r.tplIdentifier)
+ backupSchedule := &dpv1alpha1.BackupSchedule{}
+ if err := r.Client.Get(r.Context, client.ObjectKey{
+ Namespace: cluster.Namespace,
+ Name: scheduleName,
+ }, backupSchedule); client.IgnoreNotFound(err) != nil {
+ return nil, nil
+ }
+
+ if len(backupSchedule.Name) == 0 {
+ // build a new backup schedule from the backup policy template.
+ return r.buildBackupSchedule(scheduleName, backupPolicy), ictrltypes.ActionCreatePtr()
+ }
+ return backupSchedule, nil
+}
+
+func (r *BackupPolicyTplTransformer) buildBackupSchedule(
+ name string,
+ backupPolicy *dpv1alpha1.BackupPolicy) *dpv1alpha1.BackupSchedule {
+ cluster := r.OrigCluster
+ backupSchedule := &dpv1alpha1.BackupSchedule{
+ ObjectMeta: metav1.ObjectMeta{
+ Name: name,
+ Namespace: cluster.Namespace,
+ Labels: r.buildLabels(),
+ Annotations: r.buildAnnotations(),
+ },
+ Spec: dpv1alpha1.BackupScheduleSpec{
+ BackupPolicyName: backupPolicy.Name,
+ },
+ }
+
+ var schedules []dpv1alpha1.SchedulePolicy
+ for _, s := range r.backupPolicy.Schedules {
+ schedules = append(schedules, dpv1alpha1.SchedulePolicy{
+ BackupMethod: s.BackupMethod,
+ CronExpression: s.CronExpression,
+ Enabled: s.Enabled,
+ RetentionPeriod: r.backupPolicy.RetentionPeriod,
+ })
+ }
+ backupSchedule.Spec.Schedules = schedules
+ return backupSchedule
+}
+
// syncBackupPolicy syncs labels and annotations of the backup policy with the cluster changes.
-func (r *BackupPolicyTPLTransformer) syncBackupPolicy(backupPolicy *dataprotectionv1alpha1.BackupPolicy,
- cluster *appsv1alpha1.Cluster,
- policyTPL appsv1alpha1.BackupPolicy,
- workloadType appsv1alpha1.WorkloadType,
- tpl *appsv1alpha1.BackupPolicyTemplate) {
+func (r *BackupPolicyTplTransformer) syncBackupPolicy(backupPolicy *dpv1alpha1.BackupPolicy) {
// update labels and annotations of the backup policy.
if backupPolicy.Annotations == nil {
backupPolicy.Annotations = map[string]string{}
}
- backupPolicy.Annotations[constant.DefaultBackupPolicyAnnotationKey] = r.defaultPolicyAnnotationValue()
- backupPolicy.Annotations[constant.BackupPolicyTemplateAnnotationKey] = tpl.Name
- if tpl.Annotations[constant.ReconfigureRefAnnotationKey] != "" {
- backupPolicy.Annotations[constant.ReconfigureRefAnnotationKey] = tpl.Annotations[constant.ReconfigureRefAnnotationKey]
- }
if backupPolicy.Labels == nil {
backupPolicy.Labels = map[string]string{}
}
- backupPolicy.Labels[constant.AppInstanceLabelKey] = cluster.Name
- backupPolicy.Labels[constant.KBAppComponentDefRefLabelKey] = policyTPL.ComponentDefRef
- backupPolicy.Labels[constant.AppManagedByLabelKey] = constant.AppName
+ mergeMap(backupPolicy.Annotations, r.buildAnnotations())
+ mergeMap(backupPolicy.Labels, r.buildLabels())
- // only update the role labelSelector of the backup target instance when component workload is Replication/Consensus.
- // because the replicas of component will change, such as 2->1. then if the target role is 'follower' and replicas is 1,
- // the target instance can not be found. so we sync the label selector automatically.
- if !slices.Contains([]appsv1alpha1.WorkloadType{appsv1alpha1.Replication, appsv1alpha1.Consensus}, workloadType) {
+ // only update the role labelSelector of the backup target instance when
+ // component workload is Replication/Consensus. Because the replicas of
+ // component will change, such as 2->1. then if the target role is 'follower'
+ // and replicas is 1, the target instance can not be found. so we sync the
+ // label selector automatically.
+ if !workloadHasRoleLabel(r.compWorkloadType) {
return
}
- component := r.getFirstComponent(cluster, policyTPL.ComponentDefRef)
- if component == nil {
+
+ comp := r.getClusterComponentSpec()
+ if comp == nil {
return
}
+
// convert role labelSelector based on the replicas of the component automatically.
- syncTheRoleLabel := func(target dataprotectionv1alpha1.TargetCluster,
- basePolicy appsv1alpha1.BasePolicy) dataprotectionv1alpha1.TargetCluster {
- role := basePolicy.Target.Role
- if len(role) == 0 {
- return target
- }
- if target.LabelsSelector == nil || target.LabelsSelector.MatchLabels == nil {
- target.LabelsSelector = &metav1.LabelSelector{MatchLabels: map[string]string{}}
- }
- if component.Replicas == 1 {
- // if replicas is 1, remove the role label selector.
- delete(target.LabelsSelector.MatchLabels, constant.RoleLabelKey)
- } else {
- target.LabelsSelector.MatchLabels[constant.RoleLabelKey] = role
- }
- return target
+ // TODO(ldm): need more review.
+ role := r.backupPolicy.Target.Role
+ if len(role) == 0 {
+ return
}
- if backupPolicy.Spec.Snapshot != nil && policyTPL.Snapshot != nil {
- backupPolicy.Spec.Snapshot.Target = syncTheRoleLabel(backupPolicy.Spec.Snapshot.Target,
- policyTPL.Snapshot.BasePolicy)
+
+ podSelector := backupPolicy.Spec.Target.PodSelector
+ if podSelector.LabelSelector == nil || podSelector.LabelSelector.MatchLabels == nil {
+ podSelector.LabelSelector = &metav1.LabelSelector{MatchLabels: map[string]string{}}
}
- if backupPolicy.Spec.Datafile != nil && policyTPL.Datafile != nil {
- backupPolicy.Spec.Datafile.Target = syncTheRoleLabel(backupPolicy.Spec.Datafile.Target,
- policyTPL.Datafile.BasePolicy)
+ if r.getCompReplicas() == 1 {
+ delete(podSelector.LabelSelector.MatchLabels, constant.RoleLabelKey)
+ } else {
+ podSelector.LabelSelector.MatchLabels[constant.RoleLabelKey] = role
}
- if backupPolicy.Spec.Logfile != nil && policyTPL.Logfile != nil {
- backupPolicy.Spec.Logfile.Target = syncTheRoleLabel(backupPolicy.Spec.Logfile.Target,
- policyTPL.Logfile.BasePolicy)
+}
+
+func (r *BackupPolicyTplTransformer) getCompReplicas() int32 {
+ rsm := &workloads.ReplicatedStateMachine{}
+ compSpec := r.getClusterComponentSpec()
+ rsmName := fmt.Sprintf("%s-%s", r.Cluster.Name, compSpec.Name)
+ if err := r.Client.Get(r.Context, client.ObjectKey{Name: rsmName, Namespace: r.Cluster.Namespace}, rsm); err != nil {
+ return compSpec.Replicas
}
+ return *rsm.Spec.Replicas
}
-// buildBackupPolicy builds a new backup policy from the backup policy template.
-func (r *BackupPolicyTPLTransformer) buildBackupPolicy(policyTPL appsv1alpha1.BackupPolicy,
- cluster *appsv1alpha1.Cluster,
- workloadType appsv1alpha1.WorkloadType,
- tpl *appsv1alpha1.BackupPolicyTemplate,
- backupPolicyName string) *dataprotectionv1alpha1.BackupPolicy {
- component := r.getFirstComponent(cluster, policyTPL.ComponentDefRef)
- if component == nil {
+// buildBackupPolicy builds a new backup policy by the backup policy template.
+func (r *BackupPolicyTplTransformer) buildBackupPolicy(backupPolicyName string) *dpv1alpha1.BackupPolicy {
+ comp := r.getClusterComponentSpec()
+ if comp == nil {
return nil
}
- backupPolicy := &dataprotectionv1alpha1.BackupPolicy{
+ cluster := r.OrigCluster
+ backupPolicy := &dpv1alpha1.BackupPolicy{
ObjectMeta: metav1.ObjectMeta{
- Name: backupPolicyName,
- Namespace: cluster.Namespace,
- Labels: map[string]string{
- constant.AppInstanceLabelKey: cluster.Name,
- constant.KBAppComponentDefRefLabelKey: policyTPL.ComponentDefRef,
- constant.AppManagedByLabelKey: constant.AppName,
- },
- Annotations: map[string]string{
- constant.DefaultBackupPolicyAnnotationKey: r.defaultPolicyAnnotationValue(),
- constant.BackupPolicyTemplateAnnotationKey: tpl.Name,
- constant.BackupDataPathPrefixAnnotationKey: fmt.Sprintf("/%s-%s/%s", cluster.Name, cluster.UID, component.Name),
- },
+ Name: backupPolicyName,
+ Namespace: cluster.Namespace,
+ Labels: r.buildLabels(),
+ Annotations: r.buildAnnotations(),
},
}
- if tpl.Annotations[constant.ReconfigureRefAnnotationKey] != "" {
- backupPolicy.Annotations[constant.ReconfigureRefAnnotationKey] = tpl.Annotations[constant.ReconfigureRefAnnotationKey]
- }
+
bpSpec := backupPolicy.Spec
- if policyTPL.Retention != nil {
- bpSpec.Retention = &dataprotectionv1alpha1.RetentionSpec{
- TTL: policyTPL.Retention.TTL,
- }
- }
- bpSpec.Schedule.StartingDeadlineMinutes = policyTPL.Schedule.StartingDeadlineMinutes
- bpSpec.Schedule.Snapshot = r.convertSchedulePolicy(policyTPL.Schedule.Snapshot)
- bpSpec.Schedule.Datafile = r.convertSchedulePolicy(policyTPL.Schedule.Datafile)
- bpSpec.Schedule.Logfile = r.convertSchedulePolicy(policyTPL.Schedule.Logfile)
- bpSpec.Datafile = r.convertCommonPolicy(policyTPL.Datafile, cluster.Name, *component, workloadType)
- bpSpec.Logfile = r.convertCommonPolicy(policyTPL.Logfile, cluster.Name, *component, workloadType)
- bpSpec.Snapshot = r.convertSnapshotPolicy(policyTPL.Snapshot, cluster.Name, *component, workloadType)
+ bpSpec.BackupMethods = r.backupPolicy.BackupMethods
+ bpSpec.PathPrefix = buildBackupPathPrefix(cluster, comp.Name)
+ bpSpec.Target = r.buildBackupTarget(comp)
backupPolicy.Spec = bpSpec
return backupPolicy
}
-// mergeClusterBackup merges the cluster backup configuration into the backup policy.
-func (r *BackupPolicyTPLTransformer) mergeClusterBackup(transCtx *ClusterTransformContext, cluster *appsv1alpha1.Cluster,
- backupPolicy *dataprotectionv1alpha1.BackupPolicy) {
+func (r *BackupPolicyTplTransformer) buildBackupTarget(
+ comp *appsv1alpha1.ClusterComponentSpec) *dpv1alpha1.BackupTarget {
+ targetTpl := r.backupPolicy.Target
+ clusterName := r.OrigCluster.Name
+
+ getSAName := func() string {
+ if comp.ServiceAccountName != "" {
+ return comp.ServiceAccountName
+ }
+ return "kb-" + r.Cluster.Name
+ }
+ // build the target connection credential
+ cc := dpv1alpha1.ConnectionCredential{}
+ if len(targetTpl.Account) > 0 {
+ cc.SecretName = fmt.Sprintf("%s-%s-%s", clusterName, comp.Name, targetTpl.Account)
+ cc.PasswordKey = constant.AccountPasswdForSecret
+ cc.PasswordKey = constant.AccountNameForSecret
+ } else {
+ cc.SecretName = fmt.Sprintf("%s-conn-credential", clusterName)
+ ccKey := targetTpl.ConnectionCredentialKey
+ if ccKey.PasswordKey != nil {
+ cc.PasswordKey = *ccKey.PasswordKey
+ }
+ if ccKey.UsernameKey != nil {
+ cc.UsernameKey = *ccKey.UsernameKey
+ }
+ if ccKey.PortKey != nil {
+ cc.PortKey = *ccKey.PortKey
+ }
+ if ccKey.HostKey != nil {
+ cc.HostKey = *ccKey.HostKey
+ }
+ }
+
+ target := &dpv1alpha1.BackupTarget{
+ PodSelector: &dpv1alpha1.PodSelector{
+ Strategy: dpv1alpha1.PodSelectionStrategyAny,
+ LabelSelector: &metav1.LabelSelector{
+ MatchLabels: r.buildTargetPodLabels(comp),
+ },
+ },
+ ConnectionCredential: &cc,
+ ServiceAccountName: getSAName(),
+ }
+ return target
+}
+
+func (r *BackupPolicyTplTransformer) mergeClusterBackup(
+ backupPolicy *dpv1alpha1.BackupPolicy,
+ backupSchedule *dpv1alpha1.BackupSchedule) *dpv1alpha1.BackupSchedule {
+ cluster := r.OrigCluster
backupEnabled := func() bool {
return cluster.Spec.Backup != nil && boolValue(cluster.Spec.Backup.Enabled)
}
@@ -238,243 +385,142 @@ func (r *BackupPolicyTPLTransformer) mergeClusterBackup(transCtx *ClusterTransfo
if backupPolicy == nil || cluster.Spec.Backup == nil {
// backup policy is nil, can not enable cluster backup, so record event and return.
if backupEnabled() {
- transCtx.EventRecorder.Event(transCtx.Cluster, corev1.EventTypeWarning,
+ r.EventRecorder.Event(r.Cluster, corev1.EventTypeWarning,
"BackupPolicyNotFound", "backup policy is nil, can not enable cluster backup")
}
- return
+ return nil
}
backup := cluster.Spec.Backup
- spec := &backupPolicy.Spec
- setSchedulePolicy := func(schedulePolicy *dataprotectionv1alpha1.SchedulePolicy, enable bool) {
- if schedulePolicy == nil {
- if enable {
- // failed to find the schedule policy for backup method, so record event and return.
- transCtx.EventRecorder.Eventf(transCtx.Cluster, corev1.EventTypeWarning, "BackupSchedulePolicyNotFound",
- "failed to find the schedule policy for backup method %s", backup.Method)
- }
- return
- }
- schedulePolicy.Enable = enable
- if enable && backup.CronExpression != "" {
- schedulePolicy.CronExpression = backup.CronExpression
- }
- }
-
- // disable automated backup, set all backup schedule to false
- if !backupEnabled() {
- setSchedulePolicy(spec.Schedule.Snapshot, false)
- setSchedulePolicy(spec.Schedule.Datafile, false)
- setSchedulePolicy(spec.Schedule.Logfile, false)
- return
- }
-
- if backup.RetentionPeriod != nil {
- spec.Retention = &dataprotectionv1alpha1.RetentionSpec{
- TTL: backup.RetentionPeriod,
+ // there is no backup schedule created by backup policy template, so we need to
+ // create a new backup schedule for cluster backup.
+ if backupSchedule == nil {
+ backupSchedule = &dpv1alpha1.BackupSchedule{
+ ObjectMeta: metav1.ObjectMeta{
+ Name: generateBackupScheduleName(cluster.Name, r.backupPolicy.ComponentDefRef, r.tplIdentifier),
+ Namespace: cluster.Namespace,
+ Labels: r.buildLabels(),
+ Annotations: r.buildAnnotations(),
+ },
+ Spec: dpv1alpha1.BackupScheduleSpec{
+ BackupPolicyName: backupPolicy.Name,
+ StartingDeadlineMinutes: backup.StartingDeadlineMinutes,
+ Schedules: []dpv1alpha1.SchedulePolicy{},
+ },
}
}
- if backup.StartingDeadlineMinutes != nil {
- spec.Schedule.StartingDeadlineMinutes = backup.StartingDeadlineMinutes
- }
-
- var commonBackupPolicy *dataprotectionv1alpha1.CommonBackupPolicy
- switch backup.Method {
- case dataprotectionv1alpha1.BackupMethodSnapshot:
- // enable snapshot and disable datafile
- setSchedulePolicy(spec.Schedule.Snapshot, true)
- setSchedulePolicy(spec.Schedule.Datafile, false)
- case dataprotectionv1alpha1.BackupMethodBackupTool:
- // disable snapshot and enable datafile
- setSchedulePolicy(spec.Schedule.Snapshot, false)
- setSchedulePolicy(spec.Schedule.Datafile, true)
- commonBackupPolicy = spec.Datafile
+ // build backup schedule policy by cluster backup spec
+ sp := &dpv1alpha1.SchedulePolicy{
+ Enabled: backup.Enabled,
+ RetentionPeriod: backup.RetentionPeriod,
+ BackupMethod: backup.Method,
+ CronExpression: backup.CronExpression,
}
- setRepoName := func(bp *dataprotectionv1alpha1.CommonBackupPolicy) {
- if backup.RepoName == "" || bp == nil {
- return
+ // merge cluster backup schedule policy into backup schedule, if the backup
+ // schedule with specified method already exists, we need to update it
+ // using the cluster backup schedule policy. Otherwise, we need to append
+ // it to the backup schedule.
+ for i, s := range backupSchedule.Spec.Schedules {
+ if s.BackupMethod == backup.Method {
+ mergeSchedulePolicy(sp, &backupSchedule.Spec.Schedules[i])
+ return backupSchedule
}
- bp.BackupRepoName = &backup.RepoName
- }
- setRepoName(commonBackupPolicy)
- setRepoName(spec.Logfile)
-
- pitrEnabled := boolValue(backup.PITREnabled)
- if backupPolicy.Spec.Schedule.Logfile != nil {
- backupPolicy.Spec.Schedule.Logfile.Enable = pitrEnabled
- } else if pitrEnabled {
- // TODO: if backupPolicy.Spec.Schedule.Logfile is nil, and backup.PITREnabled is true,
- // should we create a new SchedulePolicy for logfile?
- // Now, hscale also maintains a backupPolicy, we can not distinguish the backupPolicy for backup
- // or hscale, so we can not create a new SchedulePolicy for logfile.
- // Need a method to distinguish the backupPolicy for backup or hscale in the future.
- transCtx.EventRecorder.Eventf(transCtx.Cluster, corev1.EventTypeWarning,
- "BackupSchedulePolicyNotFound", "failed to find the schedule policy for PITR")
}
+ backupSchedule.Spec.Schedules = append(backupSchedule.Spec.Schedules, *sp)
+ return backupSchedule
}
-// getFirstComponent returns the first component name of the componentDefRef.
-func (r *BackupPolicyTPLTransformer) getFirstComponent(cluster *appsv1alpha1.Cluster,
- componentDefRef string) *appsv1alpha1.ClusterComponentSpec {
- for _, v := range cluster.Spec.ComponentSpecs {
- if v.ComponentDefRef == componentDefRef {
+// getClusterComponentSpec returns the first component name of the componentDefRef.
+func (r *BackupPolicyTplTransformer) getClusterComponentSpec() *appsv1alpha1.ClusterComponentSpec {
+ for _, v := range r.OrigCluster.Spec.ComponentSpecs {
+ if v.ComponentDefRef == r.backupPolicy.ComponentDefRef {
return &v
}
}
return nil
}
-// convertSchedulePolicy converts the schedulePolicy from backupPolicyTemplate.
-func (r *BackupPolicyTPLTransformer) convertSchedulePolicy(sp *appsv1alpha1.SchedulePolicy) *dataprotectionv1alpha1.SchedulePolicy {
- if sp == nil {
- return nil
- }
- return &dataprotectionv1alpha1.SchedulePolicy{
- Enable: sp.Enable,
- CronExpression: sp.CronExpression,
+func (r *BackupPolicyTplTransformer) defaultPolicyAnnotationValue() string {
+ if r.tplCount > 1 && r.isDefaultTemplate != trueVal {
+ return "false"
}
+ return trueVal
}
-// convertBasePolicy converts the basePolicy from backupPolicyTemplate.
-func (r *BackupPolicyTPLTransformer) convertBasePolicy(bp appsv1alpha1.BasePolicy,
- clusterName string,
- component appsv1alpha1.ClusterComponentSpec,
- workloadType appsv1alpha1.WorkloadType) dataprotectionv1alpha1.BasePolicy {
- basePolicy := dataprotectionv1alpha1.BasePolicy{
- Target: dataprotectionv1alpha1.TargetCluster{
- LabelsSelector: &metav1.LabelSelector{
- MatchLabels: map[string]string{
- constant.AppInstanceLabelKey: clusterName,
- constant.KBAppComponentLabelKey: component.Name,
- constant.AppManagedByLabelKey: constant.AppName,
- },
- },
- },
- BackupsHistoryLimit: bp.BackupsHistoryLimit,
- OnFailAttempted: bp.OnFailAttempted,
- }
- if len(bp.BackupStatusUpdates) != 0 {
- backupStatusUpdates := make([]dataprotectionv1alpha1.BackupStatusUpdate, len(bp.BackupStatusUpdates))
- for i, v := range bp.BackupStatusUpdates {
- backupStatusUpdates[i] = dataprotectionv1alpha1.BackupStatusUpdate{
- Path: v.Path,
- ContainerName: v.ContainerName,
- Script: v.Script,
- UseTargetPodServiceAccount: v.UseTargetPodServiceAccount,
- UpdateStage: dataprotectionv1alpha1.BackupStatusUpdateStage(v.UpdateStage),
- }
- }
- basePolicy.BackupStatusUpdates = backupStatusUpdates
- }
- switch workloadType {
- case appsv1alpha1.Replication, appsv1alpha1.Consensus:
- if len(bp.Target.Role) > 0 && component.Replicas > 1 {
- // the role only works when the component has multiple replicas.
- basePolicy.Target.LabelsSelector.MatchLabels[constant.RoleLabelKey] = bp.Target.Role
- }
+func (r *BackupPolicyTplTransformer) buildAnnotations() map[string]string {
+ annotations := map[string]string{
+ dptypes.DefaultBackupPolicyAnnotationKey: r.defaultPolicyAnnotationValue(),
+ constant.BackupPolicyTemplateAnnotationKey: r.backupPolicyTpl.Name,
}
- // build the target secret.
- if len(bp.Target.Account) > 0 {
- basePolicy.Target.Secret = &dataprotectionv1alpha1.BackupPolicySecret{
- Name: fmt.Sprintf("%s-%s-%s", clusterName, component.Name, bp.Target.Account),
- PasswordKey: constant.AccountPasswdForSecret,
- UsernameKey: constant.AccountNameForSecret,
- }
- } else {
- basePolicy.Target.Secret = &dataprotectionv1alpha1.BackupPolicySecret{
- Name: fmt.Sprintf("%s-conn-credential", clusterName),
- }
- connectionCredentialKey := bp.Target.ConnectionCredentialKey
- if connectionCredentialKey.PasswordKey != nil {
- basePolicy.Target.Secret.PasswordKey = *connectionCredentialKey.PasswordKey
- }
- if connectionCredentialKey.UsernameKey != nil {
- basePolicy.Target.Secret.UsernameKey = *connectionCredentialKey.UsernameKey
- }
+ if r.backupPolicyTpl.Annotations[dptypes.ReconfigureRefAnnotationKey] != "" {
+ annotations[dptypes.ReconfigureRefAnnotationKey] = r.backupPolicyTpl.Annotations[dptypes.ReconfigureRefAnnotationKey]
}
- return basePolicy
+ return annotations
}
-// convertBaseBackupSchedulePolicy converts the snapshotPolicy from backupPolicyTemplate.
-func (r *BackupPolicyTPLTransformer) convertSnapshotPolicy(sp *appsv1alpha1.SnapshotPolicy,
- clusterName string,
- component appsv1alpha1.ClusterComponentSpec,
- workloadType appsv1alpha1.WorkloadType) *dataprotectionv1alpha1.SnapshotPolicy {
- if sp == nil {
- return nil
- }
- snapshotPolicy := &dataprotectionv1alpha1.SnapshotPolicy{
- BasePolicy: r.convertBasePolicy(sp.BasePolicy, clusterName, component, workloadType),
+func (r *BackupPolicyTplTransformer) buildLabels() map[string]string {
+ return map[string]string{
+ constant.AppInstanceLabelKey: r.OrigCluster.Name,
+ constant.KBAppComponentDefRefLabelKey: r.backupPolicy.ComponentDefRef,
+ constant.AppManagedByLabelKey: constant.AppName,
}
- if sp.Hooks != nil {
- snapshotPolicy.Hooks = &dataprotectionv1alpha1.BackupPolicyHook{
- PreCommands: sp.Hooks.PreCommands,
- PostCommands: sp.Hooks.PostCommands,
- ContainerName: sp.Hooks.ContainerName,
- Image: sp.Hooks.Image,
- }
- }
- return snapshotPolicy
}
-// convertBaseBackupSchedulePolicy converts the commonPolicy from backupPolicyTemplate.
-func (r *BackupPolicyTPLTransformer) convertCommonPolicy(bp *appsv1alpha1.CommonBackupPolicy,
- clusterName string,
- component appsv1alpha1.ClusterComponentSpec,
- workloadType appsv1alpha1.WorkloadType) *dataprotectionv1alpha1.CommonBackupPolicy {
- if bp == nil {
- return nil
- }
- defaultCreatePolicy := dataprotectionv1alpha1.CreatePVCPolicyIfNotPresent
- globalCreatePolicy := viper.GetString(constant.CfgKeyBackupPVCCreatePolicy)
- if dataprotectionv1alpha1.CreatePVCPolicy(globalCreatePolicy) == dataprotectionv1alpha1.CreatePVCPolicyNever {
- defaultCreatePolicy = dataprotectionv1alpha1.CreatePVCPolicyNever
- }
- defaultInitCapacity := constant.DefaultBackupPvcInitCapacity
- globalInitCapacity := viper.GetString(constant.CfgKeyBackupPVCInitCapacity)
- if len(globalInitCapacity) != 0 {
- defaultInitCapacity = globalInitCapacity
- }
- // set the persistent volume configmap infos if these variables exist.
- globalPVConfigMapName := viper.GetString(constant.CfgKeyBackupPVConfigmapName)
- globalPVConfigMapNamespace := viper.GetString(constant.CfgKeyBackupPVConfigmapNamespace)
- var persistentVolumeConfigMap *dataprotectionv1alpha1.PersistentVolumeConfigMap
- if globalPVConfigMapName != "" && globalPVConfigMapNamespace != "" {
- persistentVolumeConfigMap = &dataprotectionv1alpha1.PersistentVolumeConfigMap{
- Name: globalPVConfigMapName,
- Namespace: globalPVConfigMapNamespace,
- }
- }
- globalStorageClass := viper.GetString(constant.CfgKeyBackupPVCStorageClass)
- var storageClassName *string
- if globalStorageClass != "" {
- storageClassName = &globalStorageClass
- }
- return &dataprotectionv1alpha1.CommonBackupPolicy{
- BackupToolName: bp.BackupToolName,
- PersistentVolumeClaim: dataprotectionv1alpha1.PersistentVolumeClaim{
- InitCapacity: resource.MustParse(defaultInitCapacity),
- CreatePolicy: defaultCreatePolicy,
- PersistentVolumeConfigMap: persistentVolumeConfigMap,
- StorageClassName: storageClassName,
- },
- BasePolicy: r.convertBasePolicy(bp.BasePolicy, clusterName, component, workloadType),
- }
+// buildTargetPodLabels builds the target labels for the backup policy that will be
+// used to select the target pod.
+func (r *BackupPolicyTplTransformer) buildTargetPodLabels(comp *appsv1alpha1.ClusterComponentSpec) map[string]string {
+ labels := map[string]string{
+ constant.AppInstanceLabelKey: r.OrigCluster.Name,
+ constant.KBAppComponentLabelKey: comp.Name,
+ constant.AppManagedByLabelKey: constant.AppName,
+ }
+ // append label to filter specific role of the component.
+ targetTpl := &r.backupPolicy.Target
+ if workloadHasRoleLabel(r.compWorkloadType) &&
+ len(targetTpl.Role) > 0 && r.getCompReplicas() > 1 {
+ // the role only works when the component has multiple replicas.
+ labels[constant.RoleLabelKey] = targetTpl.Role
+ }
+ return labels
}
-func (r *BackupPolicyTPLTransformer) defaultPolicyAnnotationValue() string {
- if r.tplCount > 1 && r.isDefaultTemplate != trueVal {
- return "false"
+// generateBackupPolicyName generates the backup policy name which is created from backup policy template.
+func generateBackupPolicyName(clusterName, componentDef, identifier string) string {
+ if len(identifier) == 0 {
+ return fmt.Sprintf("%s-%s-backup-policy", clusterName, componentDef)
}
- return trueVal
+ return fmt.Sprintf("%s-%s-backup-policy-%s", clusterName, componentDef, identifier)
}
-// DeriveBackupPolicyName generates the backup policy name which is created from backup policy template.
-func DeriveBackupPolicyName(clusterName, componentDef, identifier string) string {
+// generateBackupScheduleName generates the backup schedule name which is created from backup policy template.
+func generateBackupScheduleName(clusterName, componentDef, identifier string) string {
if len(identifier) == 0 {
- return fmt.Sprintf("%s-%s-backup-policy", clusterName, componentDef)
+ return fmt.Sprintf("%s-%s-backup-schedule", clusterName, componentDef)
+ }
+ return fmt.Sprintf("%s-%s-backup-schedule-%s", clusterName, componentDef, identifier)
+}
+
+func buildBackupPathPrefix(cluster *appsv1alpha1.Cluster, compName string) string {
+ return fmt.Sprintf("/%s-%s/%s", cluster.Name, cluster.UID, compName)
+}
+
+func workloadHasRoleLabel(workloadType appsv1alpha1.WorkloadType) bool {
+ return slices.Contains([]appsv1alpha1.WorkloadType{appsv1alpha1.Replication, appsv1alpha1.Consensus}, workloadType)
+}
+
+func mergeSchedulePolicy(src *dpv1alpha1.SchedulePolicy, dst *dpv1alpha1.SchedulePolicy) {
+ if src.Enabled != nil {
+ dst.Enabled = src.Enabled
+ }
+ if src.RetentionPeriod.String() != "" {
+ dst.RetentionPeriod = src.RetentionPeriod
+ }
+ if src.BackupMethod != "" {
+ dst.BackupMethod = src.BackupMethod
+ }
+ if src.CronExpression != "" {
+ dst.CronExpression = src.CronExpression
}
- return fmt.Sprintf("%s-%s-backup-policy-%s", clusterName, componentDef, identifier)
}
diff --git a/controllers/apps/transformer_cluster_credential.go b/controllers/apps/transformer_cluster_credential.go
index 6380d93a3f4..a4ae3233b7f 100644
--- a/controllers/apps/transformer_cluster_credential.go
+++ b/controllers/apps/transformer_cluster_credential.go
@@ -20,11 +20,11 @@ along with this program. If not, see .
package apps
import (
- "github.com/apecloud/kubeblocks/internal/controller/factory"
corev1 "k8s.io/api/core/v1"
"sigs.k8s.io/controller-runtime/pkg/log"
"github.com/apecloud/kubeblocks/internal/controller/component"
+ "github.com/apecloud/kubeblocks/internal/controller/factory"
"github.com/apecloud/kubeblocks/internal/controller/graph"
ictrltypes "github.com/apecloud/kubeblocks/internal/controller/types"
intctrlutil "github.com/apecloud/kubeblocks/internal/controllerutil"
@@ -73,10 +73,7 @@ func (c *ClusterCredentialTransformer) Transform(ctx graph.TransformContext, dag
}
}
if synthesizedComponent != nil {
- secret, err := factory.BuildConnCredential(transCtx.ClusterDef, cluster, synthesizedComponent)
- if err != nil {
- return err
- }
+ secret := factory.BuildConnCredential(transCtx.ClusterDef, cluster, synthesizedComponent)
if secret != nil {
ictrltypes.LifecycleObjectCreate(dag, secret, root)
}
diff --git a/controllers/apps/transformer_cluster_deletion.go b/controllers/apps/transformer_cluster_deletion.go
index c7715c8a598..2a0d62875a6 100644
--- a/controllers/apps/transformer_cluster_deletion.go
+++ b/controllers/apps/transformer_cluster_deletion.go
@@ -24,7 +24,6 @@ import (
"strings"
"time"
- appsv1 "k8s.io/api/apps/v1"
batchv1 "k8s.io/api/batch/v1"
corev1 "k8s.io/api/core/v1"
policyv1 "k8s.io/api/policy/v1"
@@ -34,12 +33,11 @@ import (
"sigs.k8s.io/controller-runtime/pkg/controller/controllerutil"
appsv1alpha1 "github.com/apecloud/kubeblocks/apis/apps/v1alpha1"
- dataprotectionv1alpha1 "github.com/apecloud/kubeblocks/apis/dataprotection/v1alpha1"
+ dpv1alpha1 "github.com/apecloud/kubeblocks/apis/dataprotection/v1alpha1"
workloads "github.com/apecloud/kubeblocks/apis/workloads/v1alpha1"
"github.com/apecloud/kubeblocks/internal/constant"
"github.com/apecloud/kubeblocks/internal/controller/graph"
ictrltypes "github.com/apecloud/kubeblocks/internal/controller/types"
- intctrlutil "github.com/apecloud/kubeblocks/internal/controllerutil"
)
// ClusterDeletionTransformer handles cluster deletion
@@ -201,11 +199,8 @@ func kindsForHalt() ([]client.ObjectList, []client.ObjectList) {
nonNamespacedKindsPlus := []client.ObjectList{
&rbacv1.ClusterRoleBindingList{},
}
- if intctrlutil.IsRSMEnabled() {
- namespacedKindsPlus = append(namespacedKindsPlus, &workloads.ReplicatedStateMachineList{})
- } else {
- namespacedKindsPlus = append(namespacedKindsPlus, &corev1.ServiceList{}, &appsv1.StatefulSetList{}, &appsv1.DeploymentList{})
- }
+ namespacedKindsPlus = append(namespacedKindsPlus, &workloads.ReplicatedStateMachineList{})
+
return append(namespacedKinds, namespacedKindsPlus...), append(nonNamespacedKinds, nonNamespacedKindsPlus...)
}
@@ -215,8 +210,10 @@ func kindsForDelete() ([]client.ObjectList, []client.ObjectList) {
&corev1.SecretList{},
&corev1.ConfigMapList{},
&corev1.PersistentVolumeClaimList{},
- &dataprotectionv1alpha1.BackupPolicyList{},
+ &dpv1alpha1.BackupPolicyList{},
+ &dpv1alpha1.BackupScheduleList{},
&batchv1.JobList{},
+ &dpv1alpha1.RestoreList{},
}
return append(namespacedKinds, namespacedKindsPlus...), nonNamespacedKinds
}
@@ -224,7 +221,7 @@ func kindsForDelete() ([]client.ObjectList, []client.ObjectList) {
func kindsForWipeOut() ([]client.ObjectList, []client.ObjectList) {
namespacedKinds, nonNamespacedKinds := kindsForDelete()
namespacedKindsPlus := []client.ObjectList{
- &dataprotectionv1alpha1.BackupList{},
+ &dpv1alpha1.BackupList{},
}
return append(namespacedKinds, namespacedKindsPlus...), nonNamespacedKinds
}
diff --git a/controllers/apps/transformer_rbac.go b/controllers/apps/transformer_rbac.go
index 6ba7e352e61..fad83c24da7 100644
--- a/controllers/apps/transformer_rbac.go
+++ b/controllers/apps/transformer_rbac.go
@@ -38,6 +38,7 @@ import (
"github.com/apecloud/kubeblocks/internal/controller/graph"
ictrltypes "github.com/apecloud/kubeblocks/internal/controller/types"
ictrlutil "github.com/apecloud/kubeblocks/internal/controllerutil"
+ dptypes "github.com/apecloud/kubeblocks/internal/dataprotection/types"
viper "github.com/apecloud/kubeblocks/internal/viperx"
)
@@ -46,12 +47,6 @@ type RBACTransformer struct{}
var _ graph.Transformer = &RBACTransformer{}
-const (
- RBACRoleName = "kubeblocks-cluster-pod-role"
- RBACClusterRoleName = "kubeblocks-volume-protection-pod-role"
- ServiceAccountKind = "ServiceAccount"
-)
-
func (c *RBACTransformer) Transform(ctx graph.TransformContext, dag *graph.DAG) error {
transCtx, _ := ctx.(*ClusterTransformContext)
cluster := transCtx.Cluster
@@ -87,18 +82,10 @@ func (c *RBACTransformer) Transform(ctx graph.TransformContext, dag *graph.DAG)
return nil
}
- rb, err := buildReloBinding(cluster, serviceAccounts)
- if err != nil {
- return err
- }
-
+ rb := buildRoleBinding(cluster, serviceAccounts)
parentVertex = ictrltypes.LifecycleObjectCreate(dag, rb, parentVertex)
if len(serviceAccountsNeedCrb) > 0 {
- crb, err := buildClusterReloBinding(cluster, serviceAccountsNeedCrb)
- if err != nil {
- return err
- }
-
+ crb := buildClusterRoleBinding(cluster, serviceAccountsNeedCrb)
parentVertex = ictrltypes.LifecycleObjectCreate(dag, crb, parentVertex)
}
@@ -190,14 +177,14 @@ func isClusterRoleBindingExist(transCtx *ClusterTransformContext, serviceAccount
return false
}
- if crb.RoleRef.Name != RBACClusterRoleName {
+ if crb.RoleRef.Name != constant.RBACClusterRoleName {
transCtx.Logger.V(1).Info("rbac manager: ClusterRole not match", "ClusterRole",
- RBACClusterRoleName, "clusterrolebinding.RoleRef", crb.RoleRef.Name)
+ constant.RBACClusterRoleName, "clusterrolebinding.RoleRef", crb.RoleRef.Name)
}
isServiceAccountMatch := false
for _, sub := range crb.Subjects {
- if sub.Kind == ServiceAccountKind && sub.Name == serviceAccountName {
+ if sub.Kind == rbacv1.ServiceAccountKind && sub.Name == serviceAccountName {
isServiceAccountMatch = true
break
}
@@ -228,14 +215,14 @@ func isRoleBindingExist(transCtx *ClusterTransformContext, serviceAccountName st
return false
}
- if rb.RoleRef.Name != RBACClusterRoleName {
+ if rb.RoleRef.Name != constant.RBACClusterRoleName {
transCtx.Logger.V(1).Info("rbac manager: ClusterRole not match", "ClusterRole",
- RBACRoleName, "rolebinding.RoleRef", rb.RoleRef.Name)
+ constant.RBACRoleName, "rolebinding.RoleRef", rb.RoleRef.Name)
}
isServiceAccountMatch := false
for _, sub := range rb.Subjects {
- if sub.Kind == ServiceAccountKind && sub.Name == serviceAccountName {
+ if sub.Kind == rbacv1.ServiceAccountKind && sub.Name == serviceAccountName {
isServiceAccountMatch = true
break
}
@@ -287,7 +274,7 @@ func getDefaultBackupPolicyTemplate(transCtx *ClusterTransformContext, clusterDe
return nil, nil
}
for _, item := range backupPolicyTPLs.Items {
- if item.Annotations[constant.DefaultBackupPolicyTemplateAnnotationKey] == trueVal {
+ if item.Annotations[dptypes.DefaultBackupPolicyTemplateAnnotationKey] == trueVal {
return &item, nil
}
}
@@ -321,11 +308,8 @@ func buildServiceAccounts(transCtx *ClusterTransformContext, componentSpecs []ap
if _, ok := serviceAccounts[serviceAccountName]; ok {
continue
}
- serviceAccount, err := factory.BuildServiceAccount(cluster)
+ serviceAccount := factory.BuildServiceAccount(cluster)
serviceAccount.Name = serviceAccountName
- if err != nil {
- return nil, nil, err
- }
serviceAccounts[serviceAccountName] = serviceAccount
if isVolumeProtectionEnabled(clusterDef, &compSpec) {
@@ -335,46 +319,40 @@ func buildServiceAccounts(transCtx *ClusterTransformContext, componentSpecs []ap
return serviceAccounts, serviceAccountsNeedCrb, nil
}
-func buildReloBinding(cluster *appsv1alpha1.Cluster, serviceAccounts map[string]*corev1.ServiceAccount) (*rbacv1.RoleBinding, error) {
- roleBinding, err := factory.BuildRoleBinding(cluster)
- if err != nil {
- return nil, err
- }
+func buildRoleBinding(cluster *appsv1alpha1.Cluster, serviceAccounts map[string]*corev1.ServiceAccount) *rbacv1.RoleBinding {
+ roleBinding := factory.BuildRoleBinding(cluster)
roleBinding.Subjects = []rbacv1.Subject{}
for saName := range serviceAccounts {
subject := rbacv1.Subject{
Name: saName,
Namespace: cluster.Namespace,
- Kind: "ServiceAccount",
+ Kind: rbacv1.ServiceAccountKind,
}
roleBinding.Subjects = append(roleBinding.Subjects, subject)
}
- return roleBinding, nil
+ return roleBinding
}
-func buildClusterReloBinding(cluster *appsv1alpha1.Cluster, serviceAccounts map[string]*corev1.ServiceAccount) (*rbacv1.ClusterRoleBinding, error) {
- clusterRoleBinding, err := factory.BuildClusterRoleBinding(cluster)
- if err != nil {
- return nil, err
- }
+func buildClusterRoleBinding(cluster *appsv1alpha1.Cluster, serviceAccounts map[string]*corev1.ServiceAccount) *rbacv1.ClusterRoleBinding {
+ clusterRoleBinding := factory.BuildClusterRoleBinding(cluster)
clusterRoleBinding.Subjects = []rbacv1.Subject{}
for saName := range serviceAccounts {
subject := rbacv1.Subject{
Name: saName,
Namespace: cluster.Namespace,
- Kind: "ServiceAccount",
+ Kind: rbacv1.ServiceAccountKind,
}
clusterRoleBinding.Subjects = append(clusterRoleBinding.Subjects, subject)
}
- return clusterRoleBinding, nil
+ return clusterRoleBinding
}
func createSaVertex(serviceAccounts map[string]*corev1.ServiceAccount, dag *graph.DAG, parentVertex *ictrltypes.LifecycleVertex) []*ictrltypes.LifecycleVertex {
- saVertexs := []*ictrltypes.LifecycleVertex{}
+ var saVertexes []*ictrltypes.LifecycleVertex
for _, sa := range serviceAccounts {
// serviceaccount must be created before rolebinding and clusterrolebinding
saVertex := ictrltypes.LifecycleObjectCreate(dag, sa, parentVertex)
- saVertexs = append(saVertexs, saVertex)
+ saVertexes = append(saVertexes, saVertex)
}
- return saVertexs
+ return saVertexes
}
diff --git a/controllers/apps/transformer_rbac_test.go b/controllers/apps/transformer_rbac_test.go
index ee35170fb77..8cb6332cc7d 100644
--- a/controllers/apps/transformer_rbac_test.go
+++ b/controllers/apps/transformer_rbac_test.go
@@ -110,12 +110,10 @@ var _ = Describe("object rbac transformer test.", func() {
&corev1.ServiceAccount{}, false)).Should(Succeed())
Expect(transformer.Transform(transCtx, dag)).Should(BeNil())
- serviceAccount, err := factory.BuildServiceAccount(cluster)
- Expect(err).Should(BeNil())
+ serviceAccount := factory.BuildServiceAccount(cluster)
serviceAccount.Name = serviceAccountName
- roleBinding, err := factory.BuildRoleBinding(cluster)
- Expect(err).Should(BeNil())
+ roleBinding := factory.BuildRoleBinding(cluster)
roleBinding.Subjects[0].Name = serviceAccountName
dagExpected := mockDAG(cluster)
@@ -130,16 +128,13 @@ var _ = Describe("object rbac transformer test.", func() {
&corev1.ServiceAccount{}, false)).Should(Succeed())
Expect(transformer.Transform(transCtx, dag)).Should(BeNil())
- serviceAccount, err := factory.BuildServiceAccount(cluster)
- Expect(err).Should(BeNil())
+ serviceAccount := factory.BuildServiceAccount(cluster)
serviceAccount.Name = serviceAccountName
- roleBinding, err := factory.BuildRoleBinding(cluster)
- Expect(err).Should(BeNil())
+ roleBinding := factory.BuildRoleBinding(cluster)
roleBinding.Subjects[0].Name = serviceAccountName
- clusterRoleBinding, err := factory.BuildClusterRoleBinding(cluster)
- Expect(err).Should(BeNil())
+ clusterRoleBinding := factory.BuildClusterRoleBinding(cluster)
clusterRoleBinding.Subjects[0].Name = serviceAccountName
dagExpected := mockDAG(cluster)
diff --git a/controllers/apps/utils.go b/controllers/apps/utils.go
index bbf32f4e063..d886bc3c7e2 100644
--- a/controllers/apps/utils.go
+++ b/controllers/apps/utils.go
@@ -87,3 +87,9 @@ func boolValue(b *bool) bool {
}
return *b
}
+
+func mergeMap(dst, src map[string]string) {
+ for key, val := range src {
+ dst[key] = val
+ }
+}
diff --git a/controllers/dataprotection/actionset_controller.go b/controllers/dataprotection/actionset_controller.go
new file mode 100644
index 00000000000..6d718ca9b1e
--- /dev/null
+++ b/controllers/dataprotection/actionset_controller.go
@@ -0,0 +1,103 @@
+/*
+Copyright (C) 2022-2023 ApeCloud Co., Ltd
+
+This file is part of KubeBlocks project
+
+This program is free software: you can redistribute it and/or modify
+it under the terms of the GNU Affero General Public License as published by
+the Free Software Foundation, either version 3 of the License, or
+(at your option) any later version.
+
+This program is distributed in the hope that it will be useful
+but WITHOUT ANY WARRANTY; without even the implied warranty of
+MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+GNU Affero General Public License for more details.
+
+You should have received a copy of the GNU Affero General Public License
+along with this program. If not, see .
+*/
+
+package dataprotection
+
+import (
+ "context"
+
+ "k8s.io/apimachinery/pkg/runtime"
+ "k8s.io/client-go/tools/record"
+ ctrl "sigs.k8s.io/controller-runtime"
+ "sigs.k8s.io/controller-runtime/pkg/client"
+ "sigs.k8s.io/controller-runtime/pkg/log"
+
+ dpv1alpha1 "github.com/apecloud/kubeblocks/apis/dataprotection/v1alpha1"
+ intctrlutil "github.com/apecloud/kubeblocks/internal/controllerutil"
+ dptypes "github.com/apecloud/kubeblocks/internal/dataprotection/types"
+)
+
+// ActionSetReconciler reconciles a ActionSet object
+type ActionSetReconciler struct {
+ client.Client
+ Scheme *runtime.Scheme
+ Recorder record.EventRecorder
+}
+
+// +kubebuilder:rbac:groups=dataprotection.kubeblocks.io,resources=actionsets,verbs=get;list;watch;create;update;patch;delete
+// +kubebuilder:rbac:groups=dataprotection.kubeblocks.io,resources=actionsets/status,verbs=get;update;patch
+// +kubebuilder:rbac:groups=dataprotection.kubeblocks.io,resources=actionsets/finalizers,verbs=update
+
+// Reconcile is part of the main kubernetes reconciliation loop which aims to
+// move the current state of the actionset closer to the desired state.
+func (r *ActionSetReconciler) Reconcile(ctx context.Context, req ctrl.Request) (ctrl.Result, error) {
+ reqCtx := intctrlutil.RequestCtx{
+ Ctx: ctx,
+ Req: req,
+ Log: log.FromContext(ctx).WithValues("actionSet", req.Name),
+ Recorder: r.Recorder,
+ }
+
+ actionSet := &dpv1alpha1.ActionSet{}
+ if err := r.Client.Get(reqCtx.Ctx, reqCtx.Req.NamespacedName, actionSet); err != nil {
+ return intctrlutil.CheckedRequeueWithError(err, reqCtx.Log, "")
+ }
+
+ // handle finalizer
+ res, err := intctrlutil.HandleCRDeletion(reqCtx, r, actionSet, dptypes.DataProtectionFinalizerName,
+ func() (*ctrl.Result, error) {
+ return nil, r.deleteExternalResources(reqCtx, actionSet)
+ })
+ if res != nil {
+ return *res, err
+ }
+
+ if actionSet.Status.ObservedGeneration == actionSet.Generation &&
+ actionSet.Status.Phase.IsAvailable() {
+ return ctrl.Result{}, nil
+ }
+
+ patchStatus := func(phase dpv1alpha1.Phase, message string) error {
+ patch := client.MergeFrom(actionSet.DeepCopy())
+ actionSet.Status.Phase = phase
+ actionSet.Status.Message = message
+ actionSet.Status.ObservedGeneration = actionSet.Generation
+ return r.Client.Status().Patch(reqCtx.Ctx, actionSet, patch)
+ }
+
+ // TODO(ldm): validate actionSet
+
+ if err := patchStatus(dpv1alpha1.AvailablePhase, ""); err != nil {
+ return intctrlutil.CheckedRequeueWithError(err, reqCtx.Log, "")
+ }
+ intctrlutil.RecordCreatedEvent(r.Recorder, actionSet)
+ return ctrl.Result{}, nil
+}
+
+// SetupWithManager sets up the controller with the Manager.
+func (r *ActionSetReconciler) SetupWithManager(mgr ctrl.Manager) error {
+ return ctrl.NewControllerManagedBy(mgr).
+ For(&dpv1alpha1.ActionSet{}).Complete(r)
+}
+
+func (r *ActionSetReconciler) deleteExternalResources(
+ _ intctrlutil.RequestCtx,
+ _ *dpv1alpha1.ActionSet) error {
+ return nil
+}
diff --git a/controllers/dataprotection/actionset_controller_test.go b/controllers/dataprotection/actionset_controller_test.go
new file mode 100644
index 00000000000..e224f7da1d6
--- /dev/null
+++ b/controllers/dataprotection/actionset_controller_test.go
@@ -0,0 +1,61 @@
+/*
+Copyright (C) 2022-2023 ApeCloud Co., Ltd
+
+This file is part of KubeBlocks project
+
+This program is free software: you can redistribute it and/or modify
+it under the terms of the GNU Affero General Public License as published by
+the Free Software Foundation, either version 3 of the License, or
+(at your option) any later version.
+
+This program is distributed in the hope that it will be useful
+but WITHOUT ANY WARRANTY; without even the implied warranty of
+MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+GNU Affero General Public License for more details.
+
+You should have received a copy of the GNU Affero General Public License
+along with this program. If not, see .
+*/
+
+package dataprotection
+
+import (
+ . "github.com/onsi/ginkgo/v2"
+ . "github.com/onsi/gomega"
+
+ "sigs.k8s.io/controller-runtime/pkg/client"
+
+ intctrlutil "github.com/apecloud/kubeblocks/internal/generics"
+ testapps "github.com/apecloud/kubeblocks/internal/testutil/apps"
+ testdp "github.com/apecloud/kubeblocks/internal/testutil/dataprotection"
+)
+
+var _ = Describe("ActionSet Controller test", func() {
+ cleanEnv := func() {
+ // must wait till resources deleted and no longer existed before the testcases start,
+ // otherwise if later it needs to create some new resource objects with the same name,
+ // in race conditions, it will find the existence of old objects, resulting failure to
+ // create the new objects.
+ By("clean resources")
+
+ ml := client.HasLabels{testCtx.TestObjLabelKey}
+
+ // non-namespaced
+ testapps.ClearResourcesWithRemoveFinalizerOption(&testCtx, intctrlutil.ActionSetSignature, true, ml)
+ }
+
+ BeforeEach(func() {
+ cleanEnv()
+ })
+
+ AfterEach(func() {
+ cleanEnv()
+ })
+
+ Context("create a actionSet", func() {
+ It("should be available", func() {
+ as := testdp.NewFakeActionSet(&testCtx)
+ Expect(as).ShouldNot(BeNil())
+ })
+ })
+})
diff --git a/controllers/dataprotection/backup_controller.go b/controllers/dataprotection/backup_controller.go
index 43047a3ef29..7f5122771d1 100644
--- a/controllers/dataprotection/backup_controller.go
+++ b/controllers/dataprotection/backup_controller.go
@@ -22,29 +22,19 @@ package dataprotection
import (
"context"
"encoding/json"
- "errors"
"fmt"
- "math"
"reflect"
- "sort"
- "strconv"
- "strings"
"time"
- ctrlbuilder "github.com/apecloud/kubeblocks/internal/controller/factory"
- snapshotv1beta1 "github.com/kubernetes-csi/external-snapshotter/client/v3/apis/volumesnapshot/v1beta1"
- snapshotv1 "github.com/kubernetes-csi/external-snapshotter/client/v6/apis/volumesnapshot/v1"
- "github.com/leaanthony/debme"
+ vsv1beta1 "github.com/kubernetes-csi/external-snapshotter/client/v3/apis/volumesnapshot/v1beta1"
+ vsv1 "github.com/kubernetes-csi/external-snapshotter/client/v6/apis/volumesnapshot/v1"
appsv1 "k8s.io/api/apps/v1"
batchv1 "k8s.io/api/batch/v1"
corev1 "k8s.io/api/core/v1"
- storagev1 "k8s.io/api/storage/v1"
apierrors "k8s.io/apimachinery/pkg/api/errors"
- "k8s.io/apimachinery/pkg/api/resource"
metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
k8sruntime "k8s.io/apimachinery/pkg/runtime"
- "k8s.io/apimachinery/pkg/types"
- "k8s.io/apimachinery/pkg/util/yaml"
+ "k8s.io/client-go/rest"
"k8s.io/client-go/tools/record"
"k8s.io/utils/clock"
ctrl "sigs.k8s.io/controller-runtime"
@@ -52,36 +42,29 @@ import (
"sigs.k8s.io/controller-runtime/pkg/client"
"sigs.k8s.io/controller-runtime/pkg/controller"
"sigs.k8s.io/controller-runtime/pkg/controller/controllerutil"
- "sigs.k8s.io/controller-runtime/pkg/handler"
"sigs.k8s.io/controller-runtime/pkg/log"
- "sigs.k8s.io/controller-runtime/pkg/reconcile"
- "sigs.k8s.io/controller-runtime/pkg/source"
appsv1alpha1 "github.com/apecloud/kubeblocks/apis/apps/v1alpha1"
- dataprotectionv1alpha1 "github.com/apecloud/kubeblocks/apis/dataprotection/v1alpha1"
+ dpv1alpha1 "github.com/apecloud/kubeblocks/apis/dataprotection/v1alpha1"
"github.com/apecloud/kubeblocks/internal/constant"
"github.com/apecloud/kubeblocks/internal/controller/model"
intctrlutil "github.com/apecloud/kubeblocks/internal/controllerutil"
+ "github.com/apecloud/kubeblocks/internal/dataprotection/action"
+ dpbackup "github.com/apecloud/kubeblocks/internal/dataprotection/backup"
+ dperrors "github.com/apecloud/kubeblocks/internal/dataprotection/errors"
+ dptypes "github.com/apecloud/kubeblocks/internal/dataprotection/types"
+ dputils "github.com/apecloud/kubeblocks/internal/dataprotection/utils"
+ "github.com/apecloud/kubeblocks/internal/dataprotection/utils/boolptr"
viper "github.com/apecloud/kubeblocks/internal/viperx"
)
-const (
- backupPathBase = "/backupdata"
- deleteBackupFilesJobNamePrefix = "delete-"
-)
-
-var (
- // errBreakReconcile is not a real error, it is used to break the current reconciliation
- errBreakReconcile = errors.New("break reconcile")
-)
-
// BackupReconciler reconciles a Backup object
type BackupReconciler struct {
client.Client
- Scheme *k8sruntime.Scheme
- Recorder record.EventRecorder
- clock clock.RealClock
- snapshotCli *intctrlutil.VolumeSnapshotCompatClient
+ Scheme *k8sruntime.Scheme
+ Recorder record.EventRecorder
+ RestConfig *rest.Config
+ clock clock.RealClock
}
// +kubebuilder:rbac:groups=dataprotection.kubeblocks.io,resources=backups,verbs=get;list;watch;create;update;patch;delete
@@ -95,12 +78,8 @@ type BackupReconciler struct {
// +kubebuilder:rbac:groups=snapshot.storage.k8s.io,resources=volumesnapshotclasses/finalizers,verbs=update;patch
// Reconcile is part of the main kubernetes reconciliation loop which aims to
-// move the current state of the cluster closer to the desired state.
-//
-// For more details, check Reconcile and its Result here:
-// - https://pkg.go.dev/sigs.k8s.io/controller-runtime@v0.12.1/pkg/reconcile
+// move the current state of the backup closer to the desired state.
func (r *BackupReconciler) Reconcile(ctx context.Context, req ctrl.Request) (ctrl.Result, error) {
- // NOTES:
// setup common request context
reqCtx := intctrlutil.RequestCtx{
Ctx: ctx,
@@ -108,37 +87,35 @@ func (r *BackupReconciler) Reconcile(ctx context.Context, req ctrl.Request) (ctr
Log: log.FromContext(ctx).WithValues("backup", req.NamespacedName),
Recorder: r.Recorder,
}
- // initialize snapshotCompatClient
- r.snapshotCli = &intctrlutil.VolumeSnapshotCompatClient{
- Client: r.Client,
- Ctx: ctx,
- }
- // Get backup obj
- backup := &dataprotectionv1alpha1.Backup{}
+
+ // get backup object, and return if not found
+ backup := &dpv1alpha1.Backup{}
if err := r.Client.Get(reqCtx.Ctx, reqCtx.Req.NamespacedName, backup); err != nil {
return intctrlutil.CheckedRequeueWithError(err, reqCtx.Log, "")
}
- reqCtx.Log.V(1).Info("in Backup Reconciler:", "backup", backup.Name, "phase", backup.Status.Phase)
- // handle deletion
- res, err := r.handleBackupDeletion(reqCtx, backup)
- if res != nil {
- return *res, err
- }
+ reqCtx.Log.V(1).Info("reconcile", "backup", req.NamespacedName, "phase", backup.Status.Phase)
- switch backup.Status.Phase {
- case "", dataprotectionv1alpha1.BackupNew:
- return r.doNewPhaseAction(reqCtx, backup)
- case dataprotectionv1alpha1.BackupInProgress:
- return r.doInProgressPhaseAction(reqCtx, backup)
- case dataprotectionv1alpha1.BackupRunning:
- if err = r.doInRunningPhaseAction(reqCtx, backup); err != nil {
- sendWarningEventForError(r.Recorder, backup, err)
+ // if backup is being deleted, set backup phase to Deleting. The backup
+ // reference workloads, data and volume snapshots will be deleted by controller
+ // later when the backup status.phase is deleting.
+ if !backup.GetDeletionTimestamp().IsZero() && backup.Status.Phase != dpv1alpha1.BackupPhaseDeleting {
+ patch := client.MergeFrom(backup.DeepCopy())
+ backup.Status.Phase = dpv1alpha1.BackupPhaseDeleting
+ if err := r.Client.Status().Patch(reqCtx.Ctx, backup, patch); err != nil {
return intctrlutil.RequeueWithError(err, reqCtx.Log, "")
}
- return intctrlutil.Reconciled()
- case dataprotectionv1alpha1.BackupCompleted:
- return r.doCompletedPhaseAction(reqCtx, backup)
+ }
+
+ switch backup.Status.Phase {
+ case "", dpv1alpha1.BackupPhaseNew:
+ return r.handleNewPhase(reqCtx, backup)
+ case dpv1alpha1.BackupPhaseRunning:
+ return r.handleRunningPhase(reqCtx, backup)
+ case dpv1alpha1.BackupPhaseCompleted:
+ return r.handleCompletedPhase(reqCtx, backup)
+ case dpv1alpha1.BackupPhaseDeleting:
+ return r.handleDeletingPhase(reqCtx, backup)
default:
return intctrlutil.Reconciled()
}
@@ -146,74 +123,42 @@ func (r *BackupReconciler) Reconcile(ctx context.Context, req ctrl.Request) (ctr
// SetupWithManager sets up the controller with the Manager.
func (r *BackupReconciler) SetupWithManager(mgr ctrl.Manager) error {
-
b := ctrl.NewControllerManagedBy(mgr).
- For(&dataprotectionv1alpha1.Backup{}).
+ For(&dpv1alpha1.Backup{}).
WithOptions(controller.Options{
MaxConcurrentReconciles: viper.GetInt(maxConcurDataProtectionReconKey),
}).
- Owns(&batchv1.Job{}).
- Owns(&appsv1.StatefulSet{}).
- Watches(&source.Kind{Type: &corev1.Pod{}}, handler.EnqueueRequestsFromMapFunc(r.filterBackupPods))
-
- if viper.GetBool("VOLUMESNAPSHOT") {
- if intctrlutil.InVolumeSnapshotV1Beta1() {
- b.Owns(&snapshotv1beta1.VolumeSnapshot{}, builder.Predicates{})
- } else {
- b.Owns(&snapshotv1.VolumeSnapshot{}, builder.Predicates{})
- }
- }
-
- return b.Complete(r)
-}
+ Owns(&batchv1.Job{})
-// checkPodsOfStatefulSetHasDeleted checks if the pods of statefulSet have been deleted
-func (r *BackupReconciler) checkPodsOfStatefulSetHasDeleted(reqCtx intctrlutil.RequestCtx, backup *dataprotectionv1alpha1.Backup) (bool, error) {
- podList := &corev1.PodList{}
- if err := r.Client.List(reqCtx.Ctx, podList, client.InNamespace(reqCtx.Req.Namespace),
- client.MatchingLabels(buildBackupWorkloadsLabels(backup))); err != nil {
- return false, err
- }
- for _, pod := range podList.Items {
- for _, owner := range pod.OwnerReferences {
- // checks if the pod is owned by sts
- if owner.Kind == constant.StatefulSetKind && owner.Name == backup.Name {
- return false, nil
- }
- }
+ if intctrlutil.InVolumeSnapshotV1Beta1() {
+ b.Owns(&vsv1beta1.VolumeSnapshot{}, builder.Predicates{})
+ } else {
+ b.Owns(&vsv1.VolumeSnapshot{}, builder.Predicates{})
}
- return true, nil
+ return b.Complete(r)
}
-// handleBackupDeleting handles the Deleting phase of backup.
-func (r *BackupReconciler) handleBackupDeleting(reqCtx intctrlutil.RequestCtx, backup *dataprotectionv1alpha1.Backup) error {
- hasDeleted, err := r.checkPodsOfStatefulSetHasDeleted(reqCtx, backup)
- if err != nil {
- return err
- }
- // wait for pods of sts clean up successfully
- if !hasDeleted {
- return nil
- }
- deleteFileJob, err := r.handleDeleteBackupFiles(reqCtx, backup)
- if err != nil {
- return err
- }
+// deleteBackupFiles deletes the backup files stored in backup repository.
+func (r *BackupReconciler) deleteBackupFiles(reqCtx intctrlutil.RequestCtx, backup *dpv1alpha1.Backup) error {
deleteBackup := func() error {
// remove backup finalizers to delete it
patch := client.MergeFrom(backup.DeepCopy())
- controllerutil.RemoveFinalizer(backup, dataProtectionFinalizerName)
+ controllerutil.RemoveFinalizer(backup, dptypes.DataProtectionFinalizerName)
return r.Patch(reqCtx.Ctx, backup, patch)
}
- // if deleteFileJob is nil, do not to delete backup files
- if deleteFileJob == nil {
- return deleteBackup()
+
+ deleter := &dpbackup.Deleter{
+ RequestCtx: reqCtx,
+ Client: r.Client,
+ Scheme: r.Scheme,
}
- if containsJobCondition(deleteFileJob, batchv1.JobComplete) {
+
+ status, err := deleter.DeleteBackupFiles(backup)
+ switch status {
+ case dpbackup.DeletionStatusSucceeded:
return deleteBackup()
- }
- if containsJobCondition(deleteFileJob, batchv1.JobFailed) {
- failureReason := fmt.Sprintf(`the job "%s" for backup files deletion failed, you can delete it to re-delete the files`, deleteFileJob.Name)
+ case dpbackup.DeletionStatusFailed:
+ failureReason := err.Error()
if backup.Status.FailureReason == failureReason {
return nil
}
@@ -221,382 +166,265 @@ func (r *BackupReconciler) handleBackupDeleting(reqCtx intctrlutil.RequestCtx, b
backup.Status.FailureReason = failureReason
r.Recorder.Event(backup, corev1.EventTypeWarning, "DeleteBackupFilesFailed", failureReason)
return r.Status().Patch(reqCtx.Ctx, backup, backupPatch)
+ case dpbackup.DeletionStatusDeleting,
+ dpbackup.DeletionStatusUnknown:
+ // wait for the deletion job completed
+ return err
}
- // wait for the deletion job completed
- return nil
+ return err
}
-func (r *BackupReconciler) handleBackupDeletion(reqCtx intctrlutil.RequestCtx, backup *dataprotectionv1alpha1.Backup) (*ctrl.Result, error) {
- if backup.Status.Phase == dataprotectionv1alpha1.BackupDeleting {
- // handle deleting
- if err := r.handleBackupDeleting(reqCtx, backup); err != nil {
- return intctrlutil.ResultToP(intctrlutil.RequeueWithError(err, reqCtx.Log, ""))
- }
- return intctrlutil.ResultToP(intctrlutil.Reconciled())
- }
- if !backup.GetDeletionTimestamp().IsZero() {
- if err := r.deleteExternalResources(reqCtx, backup); err != nil {
- return intctrlutil.ResultToP(intctrlutil.RequeueWithError(err, reqCtx.Log, ""))
- }
- // backup phase to Deleting
- patch := client.MergeFrom(backup.DeepCopy())
- backup.Status.Phase = dataprotectionv1alpha1.BackupDeleting
- if err := r.Client.Status().Patch(reqCtx.Ctx, backup, patch); err != nil {
- return intctrlutil.ResultToP(intctrlutil.RequeueWithError(err, reqCtx.Log, ""))
- }
- return intctrlutil.ResultToP(intctrlutil.Reconciled())
+// handleDeletingPhase handles the deletion of backup. It will delete the backup CR
+// and the backup workload(job/statefulset).
+func (r *BackupReconciler) handleDeletingPhase(reqCtx intctrlutil.RequestCtx, backup *dpv1alpha1.Backup) (ctrl.Result, error) {
+ // if backup phase is Deleting, delete the backup reference workloads,
+ // backup data stored in backup repository and volume snapshots.
+ // TODO(ldm): if backup is being used by restore, do not delete it.
+ if err := r.deleteExternalResources(reqCtx, backup); err != nil {
+ return intctrlutil.RequeueWithError(err, reqCtx.Log, "")
}
- return nil, nil
-}
-func (r *BackupReconciler) filterBackupPods(obj client.Object) []reconcile.Request {
- labels := obj.GetLabels()
- if v, ok := labels[constant.AppManagedByLabelKey]; !ok || v != constant.AppName {
- return []reconcile.Request{}
- }
- backupName, ok := labels[constant.DataProtectionLabelBackupNameKey]
- if !ok {
- return []reconcile.Request{}
- }
- var isCreateByStatefulSet bool
- for _, v := range obj.GetOwnerReferences() {
- if v.Kind == constant.StatefulSetKind && v.Name == backupName {
- isCreateByStatefulSet = true
- break
- }
+ if backup.Spec.DeletionPolicy == dpv1alpha1.BackupDeletionPolicyRetain {
+ return intctrlutil.Reconciled()
}
- if !isCreateByStatefulSet {
- return []reconcile.Request{}
+
+ if err := r.deleteVolumeSnapshots(reqCtx, backup); err != nil {
+ return intctrlutil.RequeueWithError(err, reqCtx.Log, "")
}
- return []reconcile.Request{
- {
- NamespacedName: types.NamespacedName{
- Namespace: obj.GetNamespace(),
- Name: backupName,
- },
- },
+
+ if err := r.deleteBackupFiles(reqCtx, backup); err != nil {
+ return intctrlutil.RequeueWithError(err, reqCtx.Log, "")
}
+ return intctrlutil.Reconciled()
}
-func (r *BackupReconciler) getBackupPolicyAndValidate(
+func (r *BackupReconciler) handleNewPhase(
reqCtx intctrlutil.RequestCtx,
- backup *dataprotectionv1alpha1.Backup) (*dataprotectionv1alpha1.BackupPolicy, error) {
- // get referenced backup policy
- backupPolicy := &dataprotectionv1alpha1.BackupPolicy{}
- backupPolicyNameSpaceName := types.NamespacedName{
- Namespace: reqCtx.Req.Namespace,
- Name: backup.Spec.BackupPolicyName,
- }
- if err := r.Get(reqCtx.Ctx, backupPolicyNameSpaceName, backupPolicy); err != nil {
- return nil, err
- }
-
- if len(backupPolicy.Name) == 0 {
- return nil, intctrlutil.NewNotFound(`backup policy "%s" not found`, backupPolicyNameSpaceName)
+ backup *dpv1alpha1.Backup) (ctrl.Result, error) {
+ request, err := r.prepareBackupRequest(reqCtx, backup)
+ if err != nil {
+ return r.updateStatusIfFailed(reqCtx, backup.DeepCopy(), backup, err)
}
- // validate backup spec
- if err := backup.Spec.Validate(backupPolicy); err != nil {
- return nil, err
+ // set and patch backup object meta, including labels, annotations and finalizers
+ // if the backup object meta is changed, the backup object will be patched.
+ if patched, err := r.patchBackupObjectMeta(backup, request); err != nil {
+ return r.updateStatusIfFailed(reqCtx, backup, request.Backup, err)
+ } else if patched {
+ return intctrlutil.Reconciled()
}
- return backupPolicy, nil
-}
-func (r *BackupReconciler) validateLogfileBackupLegitimacy(backup *dataprotectionv1alpha1.Backup,
- backupPolicy *dataprotectionv1alpha1.BackupPolicy) error {
- backupType := backup.Spec.BackupType
- if backupType != dataprotectionv1alpha1.BackupTypeLogFile {
- return nil
- }
- if backup.Name != getCreatedCRNameByBackupPolicy(backupPolicy, backupType) {
- return intctrlutil.NewInvalidLogfileBackupName(backupPolicy.Name)
- }
- if backupPolicy.Spec.Schedule.Logfile == nil {
- return intctrlutil.NewBackupNotSupported(string(backupType), backupPolicy.Name)
- }
- if !backupPolicy.Spec.Schedule.Logfile.Enable {
- return intctrlutil.NewBackupScheduleDisabled(string(backupType), backupPolicy.Name)
+ // set and patch backup status
+ if err = r.patchBackupStatus(backup, request); err != nil {
+ return r.updateStatusIfFailed(reqCtx, backup, request.Backup, err)
}
- return nil
+ return intctrlutil.Reconciled()
}
-func (r *BackupReconciler) doNewPhaseAction(
+// prepareBackupRequest prepares a request for a backup, with all references to
+// other kubernetes objects, and validate them.
+func (r *BackupReconciler) prepareBackupRequest(
reqCtx intctrlutil.RequestCtx,
- backup *dataprotectionv1alpha1.Backup) (ctrl.Result, error) {
-
- patch := client.MergeFrom(backup.DeepCopy())
- // HACK/TODO: ought to move following check to validation webhook
- if backup.Spec.BackupType == dataprotectionv1alpha1.BackupTypeSnapshot && !viper.GetBool("VOLUMESNAPSHOT") {
- backup.Status.Phase = dataprotectionv1alpha1.BackupFailed
- backup.Status.FailureReason = "VolumeSnapshot feature disabled."
- if err := r.Client.Status().Patch(reqCtx.Ctx, backup, patch); err != nil {
- return intctrlutil.CheckedRequeueWithError(err, reqCtx.Log, "")
- }
- return intctrlutil.Reconciled()
+ backup *dpv1alpha1.Backup) (*dpbackup.Request, error) {
+ request := &dpbackup.Request{
+ Backup: backup.DeepCopy(),
+ RequestCtx: reqCtx,
+ Client: r.Client,
}
- backupPolicy, err := r.getBackupPolicyAndValidate(reqCtx, backup)
- if err != nil {
- return r.updateStatusIfFailed(reqCtx, backup, err)
+ if request.Annotations == nil {
+ request.Annotations = make(map[string]string)
}
- if err = r.validateLogfileBackupLegitimacy(backup, backupPolicy); err != nil {
- return r.updateStatusIfFailed(reqCtx, backup, err)
+ if request.Labels == nil {
+ request.Labels = make(map[string]string)
}
- updateLabels := map[string]string{}
-
- // TODO: get pod with matching labels to do backup.
- var targetCluster dataprotectionv1alpha1.TargetCluster
- var isStatefulSetKind bool
- if backup.Spec.BackupType == dataprotectionv1alpha1.BackupTypeSnapshot {
- targetCluster = backupPolicy.Spec.Snapshot.Target
- } else {
- commonPolicy := backupPolicy.Spec.GetCommonPolicy(backup.Spec.BackupType)
- if commonPolicy == nil {
- return r.updateStatusIfFailed(reqCtx, backup, intctrlutil.NewBackupNotSupported(string(backup.Spec.BackupType), backupPolicy.Name))
- }
- targetCluster = commonPolicy.Target
- backupTool, err := getBackupToolByName(reqCtx, r.Client, commonPolicy.BackupToolName)
- if err != nil {
- return r.updateStatusIfFailed(reqCtx, backup, intctrlutil.NewNotFound("backupTool: %s not found", commonPolicy.BackupToolName))
- }
- if err = r.buildBackupStatusForBackupTool(reqCtx, backup, backupPolicy, commonPolicy, backupTool, updateLabels); err != nil {
- if errors.Is(err, errBreakReconcile) {
- // wait for the PVC to be created
- return intctrlutil.Reconciled()
- }
- return r.updateStatusIfFailed(reqCtx, backup, err)
- }
- isStatefulSetKind = backupTool.Spec.DeployKind == dataprotectionv1alpha1.DeployKindStatefulSet
- }
- // clean cached annotations if in NEW phase
- backupCopy := backup.DeepCopy()
- if backupCopy.Annotations[dataProtectionBackupTargetPodKey] != "" {
- delete(backupCopy.Annotations, dataProtectionBackupTargetPodKey)
- }
- target, err := r.getTargetPod(reqCtx, backupCopy, targetCluster.LabelsSelector.MatchLabels)
+ backupPolicy, err := getBackupPolicyByName(reqCtx, r.Client, backup.Spec.BackupPolicyName)
if err != nil {
- return r.updateStatusIfFailed(reqCtx, backup, err)
+ return nil, err
}
- cluster := r.getCluster(reqCtx, target)
- if hasPatch, err := r.patchBackupObjectMeta(reqCtx, backup, target, cluster, updateLabels); err != nil {
- return r.updateStatusIfFailed(reqCtx, backup, err)
- } else if hasPatch {
- return intctrlutil.Reconciled()
+ targetPods, err := getTargetPods(reqCtx, r.Client,
+ backup.Annotations[dataProtectionBackupTargetPodKey], backupPolicy)
+ if err != nil || len(targetPods) == 0 {
+ return nil, fmt.Errorf("failed to get target pods by backup policy %s/%s",
+ backupPolicy.Namespace, backupPolicy.Name)
}
- // clean up failed job if backup type is logfile
- if backup.Spec.BackupType == dataprotectionv1alpha1.BackupTypeLogFile {
- if err = r.cleanupFailedJob(reqCtx, backup); err != nil {
- return intctrlutil.CheckedRequeueWithError(err, reqCtx.Log, "")
- }
+ if len(targetPods) > 1 {
+ return nil, fmt.Errorf("do not support more than one target pods")
}
- // update Phase to InProgress/Running
- if isStatefulSetKind {
- backup.Status.Phase = dataprotectionv1alpha1.BackupRunning
- } else {
- backup.Status.Phase = dataprotectionv1alpha1.BackupInProgress
- }
- backup.Status.StartTimestamp = &metav1.Time{Time: r.clock.Now().UTC()}
- if backupPolicy.Spec.Retention != nil && backupPolicy.Spec.Retention.TTL != nil {
- backup.Status.Expiration = &metav1.Time{
- Time: backup.Status.StartTimestamp.Add(dataprotectionv1alpha1.ToDuration(backupPolicy.Spec.Retention.TTL)),
- }
+ backupMethod := getBackupMethodByName(backup.Spec.BackupMethod, backupPolicy)
+ if backupMethod == nil {
+ return nil, intctrlutil.NewNotFound("backupMethod: %s not found",
+ backup.Spec.BackupMethod)
}
- if cluster != nil {
- backup.Status.SourceCluster = cluster.Name
+ // backupMethod should specify snapshotVolumes or actionSetName, if we take
+ // snapshots to back up volumes, the snapshotVolumes should be set to true
+ // and the actionSetName is not required, if we do not take snapshots to back
+ // up volumes, the actionSetName is required.
+ snapshotVolumes := boolptr.IsSetToTrue(backupMethod.SnapshotVolumes)
+ if !snapshotVolumes && backupMethod.ActionSetName == "" {
+ return nil, fmt.Errorf("backup method %s should specify snapshotVolumes or actionSetName", backupMethod.Name)
}
- if err = r.Client.Status().Patch(reqCtx.Ctx, backup, patch); err != nil {
- return intctrlutil.CheckedRequeueWithError(err, reqCtx.Log, "")
+
+ // if backup method use volume snapshots to back up, the volume snapshot
+ // feature should be enabled.
+ if snapshotVolumes && !dputils.VolumeSnapshotEnabled() {
+ return nil, fmt.Errorf("current backup method depends on volume snapshot, but volume snapshot is not enabled")
}
- return intctrlutil.Reconciled()
-}
-func (r *BackupReconciler) buildBackupStatusForBackupTool(reqCtx intctrlutil.RequestCtx,
- backup *dataprotectionv1alpha1.Backup,
- backupPolicy *dataprotectionv1alpha1.BackupPolicy,
- commonPolicy *dataprotectionv1alpha1.CommonBackupPolicy,
- backupTool *dataprotectionv1alpha1.BackupTool,
- updateLabels map[string]string) error {
- if backup.Status.Manifests == nil {
- backup.Status.Manifests = &dataprotectionv1alpha1.ManifestsStatus{}
- }
- if backup.Status.Manifests.BackupTool == nil {
- backup.Status.Manifests.BackupTool = &dataprotectionv1alpha1.BackupToolManifestsStatus{}
- }
- // handle the PVC used in this backup
- if backup.Status.PersistentVolumeClaimName == "" {
- pvcName, pvName, err := r.handlePersistentVolumeClaim(reqCtx, backup, backupPolicy.Name, commonPolicy, updateLabels)
+ if backupMethod.ActionSetName != "" {
+ actionSet, err := getActionSetByName(reqCtx, r.Client, backupMethod.ActionSetName)
if err != nil {
- return err
+ return nil, err
}
- // record volume name
- backup.Status.PersistentVolumeClaimName = pvcName
- backup.Status.Manifests.BackupTool.VolumeName = pvName
- }
- // save the backup message for restore
- backup.Status.BackupToolName = backupTool.Name
- backupDestinationPath := getBackupDestinationPath(backup, backupPolicy.Annotations[constant.BackupDataPathPrefixAnnotationKey])
- backup.Status.Manifests.BackupTool.FilePath = backupDestinationPath
-
- if backupTool.Spec.Physical.IsRelyOnLogfile() {
- if backupPolicy.Spec.Schedule.Logfile == nil || !backupPolicy.Spec.Schedule.Logfile.Enable {
- return intctrlutil.NewBackupLogfileScheduleDisabled(backupTool.Name)
+ if actionSet.Spec.BackupType != dpv1alpha1.BackupTypeFull {
+ return nil, fmt.Errorf("only support backup type Full for actionSet %s", actionSet.Name)
}
- logfileBackupName := getCreatedCRNameByBackupPolicy(backupPolicy, dataprotectionv1alpha1.BackupTypeLogFile)
- backup.Status.Manifests.BackupTool.LogFilePath = getBackupDestinationPath(&dataprotectionv1alpha1.Backup{
- ObjectMeta: metav1.ObjectMeta{Namespace: backup.Namespace, Name: logfileBackupName},
- }, backupPolicy.Annotations[constant.BackupDataPathPrefixAnnotationKey])
+ request.ActionSet = actionSet
+ }
- logFilePvcName, _, err := r.handlePersistentVolumeClaim(reqCtx, backup, backupPolicy.Name, backupPolicy.Spec.Logfile, updateLabels)
- if err != nil {
- return err
- }
- backup.Status.LogFilePersistentVolumeClaimName = logFilePvcName
+ request.BackupPolicy = backupPolicy
+ if err = r.handleBackupRepo(request); err != nil {
+ return nil, err
}
- return nil
+
+ request.BackupMethod = backupMethod
+ request.TargetPods = targetPods
+ return request, nil
}
-func (r *BackupReconciler) cleanupFailedJob(reqCtx intctrlutil.RequestCtx, backup *dataprotectionv1alpha1.Backup) error {
- jobList := batchv1.JobList{}
- if err := r.Client.List(reqCtx.Ctx, &jobList, client.InNamespace(backup.Namespace),
- client.MatchingLabels{constant.DataProtectionLabelBackupNameKey: backup.Name}); err != nil {
- return nil
+// handleBackupRepo handles the backup repo, and get the backup repo PVC. If the
+// PVC is not present, it will add a special label and wait for the backup repo
+// controller to create the PVC.
+func (r *BackupReconciler) handleBackupRepo(request *dpbackup.Request) error {
+ repo, err := r.getBackupRepo(request.Ctx, request.Backup, request.BackupPolicy)
+ if err != nil {
+ return err
}
+ request.BackupRepo = repo
- for _, job := range jobList.Items {
- if !containsJobCondition(&job, batchv1.JobFailed) {
- continue
- }
- if err := intctrlutil.BackgroundDeleteObject(r.Client, reqCtx.Ctx, &job); err != nil {
- return err
- }
- if controllerutil.ContainsFinalizer(&job, dataProtectionFinalizerName) {
- patch := client.MergeFrom(job.DeepCopy())
- controllerutil.RemoveFinalizer(&job, dataProtectionFinalizerName)
- if err := r.Patch(reqCtx.Ctx, &job, patch); err != nil {
- return err
- }
- }
+ pvcName := repo.Status.BackupPVCName
+ if pvcName == "" {
+ return dperrors.NewBackupPVCNameIsEmpty(repo.Name, request.Spec.BackupPolicyName)
}
- return nil
-}
-func (r *BackupReconciler) handlePersistentVolumeClaim(reqCtx intctrlutil.RequestCtx,
- backup *dataprotectionv1alpha1.Backup,
- backupPolicyName string,
- commonPolicy *dataprotectionv1alpha1.CommonBackupPolicy,
- updateLabels map[string]string) (pvcName string, pvName string, err error) {
- // check the PVC from the backup repo
- pvcName, pvName, err = r.handlePVCByBackupRepo(reqCtx, backup, backupPolicyName, commonPolicy, updateLabels)
- if err == nil || !errors.Is(err, errNoDefaultBackupRepo) {
- return pvcName, pvName, err
+ pvc := &corev1.PersistentVolumeClaim{}
+ pvcKey := client.ObjectKey{Namespace: request.Req.Namespace, Name: pvcName}
+ if err = r.Client.Get(request.Ctx, pvcKey, pvc); err != nil {
+ return client.IgnoreNotFound(err)
}
- // fallback to the legacy PVC field for compatibility
- if commonPolicy.PersistentVolumeClaim.Name != nil {
- pvcName = *commonPolicy.PersistentVolumeClaim.Name
+ // backupRepo PVC exists, record the PVC name
+ if err == nil {
+ request.BackupRepoPVC = pvc
}
- pvName, err = r.handlePersistentVolumeClaimLegacy(reqCtx, backup.Spec.BackupType, backupPolicyName, commonPolicy)
- return pvcName, pvName, err
+ return nil
}
-func (r *BackupReconciler) handlePVCByBackupRepo(reqCtx intctrlutil.RequestCtx,
- backup *dataprotectionv1alpha1.Backup,
- backupPolicyName string,
- commonPolicy *dataprotectionv1alpha1.CommonBackupPolicy,
- updateLabels map[string]string) (pvcName string, pvName string, err error) {
- // check the PVC from backup repo
- repo, err := r.getBackupRepo(reqCtx, backup, commonPolicy)
+func (r *BackupReconciler) patchBackupStatus(
+ original *dpv1alpha1.Backup,
+ request *dpbackup.Request) error {
+ request.Status.FormatVersion = dpbackup.FormatVersion
+ request.Status.Path = dpbackup.BuildBackupPath(request.Backup, request.BackupPolicy.Spec.PathPrefix)
+ request.Status.Target = request.BackupPolicy.Spec.Target
+ request.Status.BackupMethod = request.BackupMethod
+ request.Status.PersistentVolumeClaimName = request.BackupRepoPVC.Name
+ request.Status.BackupRepoName = request.BackupRepo.Name
+
+ // init action status
+ actions, err := request.BuildActions()
if err != nil {
- return "", "", err
- }
- pvcName = repo.Status.BackupPVCName
- if pvcName == "" {
- err = intctrlutil.NewBackupPVCNameIsEmpty(string(backup.Spec.BackupType), backupPolicyName)
- return "", "", err
+ return err
}
- pvc := &corev1.PersistentVolumeClaim{}
- err = r.Client.Get(reqCtx.Ctx, client.ObjectKey{
- Namespace: reqCtx.Req.Namespace,
- Name: pvcName,
- }, pvc)
- if err != nil && !apierrors.IsNotFound(err) {
- // error occurred
- return "", "", err
+ request.Status.Actions = make([]dpv1alpha1.ActionStatus, len(actions))
+ for i, act := range actions {
+ request.Status.Actions[i] = dpv1alpha1.ActionStatus{
+ Name: act.GetName(),
+ Phase: dpv1alpha1.ActionPhaseNew,
+ ActionType: act.Type(),
+ }
}
- if err == nil {
- // the PVC is already present, bind the backup to the repo
- updateLabels[dataProtectionBackupRepoKey] = repo.Name
- return pvcName, pvc.Spec.VolumeName, nil
- }
- // the PVC is not present
- // add a special label and wait for the backup repo controller to create the PVC.
- // we need to update the object meta immediately, because we are going to break the current reconciliation.
- _, err = r.patchBackupObjectLabels(reqCtx, backup, map[string]string{
- dataProtectionBackupRepoKey: repo.Name,
- dataProtectionNeedRepoPVCKey: trueVal,
- })
+
+ // update phase to running
+ request.Status.Phase = dpv1alpha1.BackupPhaseRunning
+ request.Status.StartTimestamp = &metav1.Time{Time: r.clock.Now().UTC()}
+
+ duration, err := original.Spec.RetentionPeriod.ToDuration()
if err != nil {
- return "", "", err
+ return fmt.Errorf("failed to parse retention period %s, %v", original.Spec.RetentionPeriod, err)
+ }
+ if original.Spec.RetentionPeriod != "" {
+ request.Status.Expiration = &metav1.Time{
+ Time: request.Status.StartTimestamp.Add(duration),
+ }
}
- return "", "", errBreakReconcile
+ return r.Client.Status().Patch(request.Ctx, request.Backup, client.MergeFrom(original))
}
-// handlePersistentVolumeClaimLegacy handles the persistent volume claim for the backup, the rules are as follows
-// - if CreatePolicy is "Never", it will check if the pvc exists. if not existed, then report an error.
-// - if CreatePolicy is "IfNotPresent" and the pvc not existed, then create the pvc automatically.
-func (r *BackupReconciler) handlePersistentVolumeClaimLegacy(reqCtx intctrlutil.RequestCtx,
- backupType dataprotectionv1alpha1.BackupType,
- backupPolicyName string,
- commonPolicy *dataprotectionv1alpha1.CommonBackupPolicy) (string, error) {
- pvcConfig := commonPolicy.PersistentVolumeClaim
- if pvcConfig.Name == nil || len(*pvcConfig.Name) == 0 {
- return "", intctrlutil.NewBackupPVCNameIsEmpty(string(backupType), backupPolicyName)
- }
- pvc := &corev1.PersistentVolumeClaim{}
- if err := r.Client.Get(reqCtx.Ctx, client.ObjectKey{Namespace: reqCtx.Req.Namespace,
- Name: *pvcConfig.Name}, pvc); err != nil && !apierrors.IsNotFound(err) {
- return "", err
- }
- if len(pvc.Name) > 0 {
- return pvc.Spec.VolumeName, nil
- }
- if pvcConfig.CreatePolicy == dataprotectionv1alpha1.CreatePVCPolicyNever {
- return "", intctrlutil.NewNotFound(`persistent volume claim "%s" not found`, *pvcConfig.Name)
- }
- if pvcConfig.PersistentVolumeConfigMap != nil &&
- (pvcConfig.StorageClassName == nil || *pvcConfig.StorageClassName == "") {
- // if the storageClassName is empty and the PersistentVolumeConfigMap is not empty,
- // create the persistentVolume with the template
- if err := r.createPersistentVolumeWithTemplate(reqCtx, backupPolicyName, &pvcConfig); err != nil {
- return "", err
+// patchBackupObjectMeta patches backup object metaObject include cluster snapshot.
+func (r *BackupReconciler) patchBackupObjectMeta(
+ original *dpv1alpha1.Backup,
+ request *dpbackup.Request) (bool, error) {
+ targetPod := request.TargetPods[0]
+
+ // get KubeBlocks cluster and set labels and annotations for backup
+ // TODO(ldm): we should remove this dependency of cluster in the future
+ cluster := getCluster(request.Ctx, r.Client, targetPod)
+ if cluster != nil {
+ if err := setClusterSnapshotAnnotation(request.Backup, cluster); err != nil {
+ return false, err
}
+ request.Labels[dptypes.DataProtectionLabelClusterUIDKey] = string(cluster.UID)
+ }
+ for _, v := range getClusterLabelKeys() {
+ request.Labels[v] = targetPod.Labels[v]
}
- return "", r.createPVCWithStorageClassName(reqCtx, backupPolicyName, pvcConfig)
+
+ request.Labels[dataProtectionBackupRepoKey] = request.BackupRepo.Name
+ request.Labels[constant.AppManagedByLabelKey] = constant.AppName
+ request.Labels[dataProtectionLabelBackupTypeKey] = request.GetBackupType()
+
+ // if the backupRepo PVC is not present, add a special label and wait for the
+ // backup repo controller to create the PVC.
+ wait := false
+ if request.BackupRepoPVC == nil {
+ request.Labels[dataProtectionWaitRepoPreparationKey] = trueVal
+ wait = true
+ }
+
+ // set annotations
+ request.Annotations[dataProtectionBackupTargetPodKey] = targetPod.Name
+
+ // set finalizer
+ controllerutil.AddFinalizer(request.Backup, dptypes.DataProtectionFinalizerName)
+
+ if reflect.DeepEqual(original.ObjectMeta, request.ObjectMeta) {
+ return wait, nil
+ }
+
+ return true, r.Client.Patch(request.Ctx, request.Backup, client.MergeFrom(original))
}
// getBackupRepo returns the backup repo specified by the backup object or the policy.
// if no backup repo specified, it will return the default one.
-func (r *BackupReconciler) getBackupRepo(
- reqCtx intctrlutil.RequestCtx,
- backup *dataprotectionv1alpha1.Backup,
- commonPolicy *dataprotectionv1alpha1.CommonBackupPolicy) (*dataprotectionv1alpha1.BackupRepo, error) {
+func (r *BackupReconciler) getBackupRepo(ctx context.Context,
+ backup *dpv1alpha1.Backup,
+ backupPolicy *dpv1alpha1.BackupPolicy) (*dpv1alpha1.BackupRepo, error) {
// use the specified backup repo
var repoName string
if val := backup.Labels[dataProtectionBackupRepoKey]; val != "" {
repoName = val
- } else if commonPolicy.BackupRepoName != nil && *commonPolicy.BackupRepoName != "" {
- repoName = *commonPolicy.BackupRepoName
+ } else if backupPolicy.Spec.BackupRepoName != nil && *backupPolicy.Spec.BackupRepoName != "" {
+ repoName = *backupPolicy.Spec.BackupRepoName
}
if repoName != "" {
- repo := &dataprotectionv1alpha1.BackupRepo{}
- err := r.Client.Get(reqCtx.Ctx, client.ObjectKey{Name: repoName}, repo)
- if err != nil {
+ repo := &dpv1alpha1.BackupRepo{}
+ if err := r.Client.Get(ctx, client.ObjectKey{Name: repoName}, repo); err != nil {
if apierrors.IsNotFound(err) {
return nil, intctrlutil.NewNotFound("backup repo %s not found", repoName)
}
@@ -605,1404 +433,195 @@ func (r *BackupReconciler) getBackupRepo(
return repo, nil
}
// fallback to use the default repo
- return getDefaultBackupRepo(reqCtx.Ctx, r.Client)
-}
-
-// createPVCWithStorageClassName creates the persistent volume claim with the storageClassName.
-func (r *BackupReconciler) createPVCWithStorageClassName(reqCtx intctrlutil.RequestCtx,
- backupPolicyName string,
- pvcConfig dataprotectionv1alpha1.PersistentVolumeClaim) error {
- pvc := &corev1.PersistentVolumeClaim{
- ObjectMeta: metav1.ObjectMeta{
- Name: *pvcConfig.Name,
- Namespace: reqCtx.Req.Namespace,
- Annotations: buildAutoCreationAnnotations(backupPolicyName),
- },
- Spec: corev1.PersistentVolumeClaimSpec{
- StorageClassName: pvcConfig.StorageClassName,
- Resources: corev1.ResourceRequirements{
- Requests: map[corev1.ResourceName]resource.Quantity{
- corev1.ResourceStorage: pvcConfig.InitCapacity,
- },
- },
- AccessModes: []corev1.PersistentVolumeAccessMode{
- corev1.ReadWriteMany,
- },
- },
- }
- err := r.Client.Create(reqCtx.Ctx, pvc)
- return client.IgnoreAlreadyExists(err)
+ return getDefaultBackupRepo(ctx, r.Client)
}
-// createPersistentVolumeWithTemplate creates the persistent volume with the template.
-func (r *BackupReconciler) createPersistentVolumeWithTemplate(reqCtx intctrlutil.RequestCtx,
- backupPolicyName string,
- pvcConfig *dataprotectionv1alpha1.PersistentVolumeClaim) error {
- pvConfig := pvcConfig.PersistentVolumeConfigMap
- configMap := &corev1.ConfigMap{}
- if err := r.Client.Get(reqCtx.Ctx, client.ObjectKey{Namespace: pvConfig.Namespace,
- Name: pvConfig.Name}, configMap); err != nil {
- return err
- }
- pvTemplate := configMap.Data[persistentVolumeTemplateKey]
- if pvTemplate == "" {
- return intctrlutil.NewBackupPVTemplateNotFound(pvConfig.Namespace, pvConfig.Name)
- }
- pvName := fmt.Sprintf("%s-%s", *pvcConfig.Name, reqCtx.Req.Namespace)
- pvTemplate = strings.ReplaceAll(pvTemplate, "$(GENERATE_NAME)", pvName)
- pv := &corev1.PersistentVolume{}
- if err := yaml.Unmarshal([]byte(pvTemplate), pv); err != nil {
- return err
- }
- pv.Name = pvName
- pv.Spec.ClaimRef = &corev1.ObjectReference{
- Namespace: reqCtx.Req.Namespace,
- Name: *pvcConfig.Name,
- }
- pv.Annotations = buildAutoCreationAnnotations(backupPolicyName)
- // set the storageClassName to empty for the persistentVolumeClaim to avoid the dynamic provisioning
- emptyStorageClassName := ""
- pvcConfig.StorageClassName = &emptyStorageClassName
- controllerutil.AddFinalizer(pv, dataProtectionFinalizerName)
- return r.Client.Create(reqCtx.Ctx, pv)
-}
-
-func (r *BackupReconciler) doInProgressPhaseAction(
+func (r *BackupReconciler) handleRunningPhase(
reqCtx intctrlutil.RequestCtx,
- backup *dataprotectionv1alpha1.Backup) (ctrl.Result, error) {
- backupPolicy, err := r.getBackupPolicyAndValidate(reqCtx, backup)
+ backup *dpv1alpha1.Backup) (ctrl.Result, error) {
+ request, err := r.prepareBackupRequest(reqCtx, backup)
if err != nil {
- return r.updateStatusIfFailed(reqCtx, backup, err)
- }
- backupDestinationPath := getBackupDestinationPath(backup, backupPolicy.Annotations[constant.BackupDataPathPrefixAnnotationKey])
- patch := client.MergeFrom(backup.DeepCopy())
- var res *ctrl.Result
- switch backup.Spec.BackupType {
- case dataprotectionv1alpha1.BackupTypeSnapshot:
- res, err = r.doSnapshotInProgressPhaseAction(reqCtx, backup, backupPolicy, backupDestinationPath)
- default:
- res, err = r.doBaseBackupInProgressPhaseAction(reqCtx, backup, backupPolicy, backupDestinationPath)
+ return r.updateStatusIfFailed(reqCtx, backup.DeepCopy(), backup, err)
}
- if res != nil {
- return *res, err
- } else if err != nil {
- return intctrlutil.CheckedRequeueWithError(err, reqCtx.Log, "")
- }
- // finally, update backup status
- r.Recorder.Event(backup, corev1.EventTypeNormal, "CreatedBackup", "Completed backup.")
- if backup.Status.CompletionTimestamp != nil {
- // round the duration to a multiple of seconds.
- duration := backup.Status.CompletionTimestamp.Sub(backup.Status.StartTimestamp.Time).Round(time.Second)
- backup.Status.Duration = &metav1.Duration{Duration: duration}
- }
- if err := r.Client.Status().Patch(reqCtx.Ctx, backup, patch); err != nil {
- return intctrlutil.CheckedRequeueWithError(err, reqCtx.Log, "")
- }
-
- return intctrlutil.Reconciled()
-}
-
-// doSnapshotInProgressPhaseAction handles for snapshot backup during in progress.
-func (r *BackupReconciler) doSnapshotInProgressPhaseAction(reqCtx intctrlutil.RequestCtx,
- backup *dataprotectionv1alpha1.Backup,
- backupPolicy *dataprotectionv1alpha1.BackupPolicy,
- backupDestinationPath string) (*ctrl.Result, error) {
- // 1. create and ensure pre-command job completed
- // 2. create and ensure volume snapshot ready
- // 3. create and ensure post-command job completed
- snapshotSpec := backupPolicy.Spec.Snapshot
- isOK, err := r.createPreCommandJobAndEnsure(reqCtx, backup, snapshotSpec)
+ // there are actions not completed, continue to handle following actions
+ actions, err := request.BuildActions()
if err != nil {
- return intctrlutil.ResultToP(r.updateStatusIfFailed(reqCtx, backup, err))
- }
- if !isOK {
- return intctrlutil.ResultToP(intctrlutil.RequeueAfter(reconcileInterval, reqCtx.Log, ""))
- }
- if err = r.createUpdatesJobs(reqCtx, backup, nil, &snapshotSpec.BasePolicy, backupDestinationPath, dataprotectionv1alpha1.PRE); err != nil {
- r.Recorder.Event(backup, corev1.EventTypeNormal, "CreatedPreUpdatesJob", err.Error())
- }
- if err = r.createVolumeSnapshot(reqCtx, backup, backupPolicy.Spec.Snapshot); err != nil {
- return intctrlutil.ResultToP(r.updateStatusIfFailed(reqCtx, backup, err))
+ return r.updateStatusIfFailed(reqCtx, backup, request.Backup, err)
}
- key := types.NamespacedName{Namespace: reqCtx.Req.Namespace, Name: backup.Name}
- isOK, snapshotTime, err := r.ensureVolumeSnapshotReady(key)
- if err != nil {
- return intctrlutil.ResultToP(r.updateStatusIfFailed(reqCtx, backup, err))
+ actionCtx := action.Context{
+ Ctx: reqCtx.Ctx,
+ Client: r.Client,
+ Recorder: r.Recorder,
+ Scheme: r.Scheme,
+ RestClientConfig: r.RestConfig,
}
- if !isOK {
- return intctrlutil.ResultToP(intctrlutil.Reconciled())
- }
- msg := fmt.Sprintf("Created volumeSnapshot %s ready.", key.Name)
- r.Recorder.Event(backup, corev1.EventTypeNormal, "CreatedVolumeSnapshot", msg)
- isOK, err = r.createPostCommandJobAndEnsure(reqCtx, backup, snapshotSpec)
- if err != nil {
- return intctrlutil.ResultToP(r.updateStatusIfFailed(reqCtx, backup, err))
- }
- if !isOK {
- return intctrlutil.ResultToP(intctrlutil.RequeueAfter(reconcileInterval, reqCtx.Log, ""))
- }
+ // check all actions status, if any action failed, update backup status to failed
+ // if all actions completed, update backup status to completed, otherwise,
+ // continue to handle following actions.
+ for i, act := range actions {
+ status, err := act.Execute(actionCtx)
+ if err != nil {
+ return r.updateStatusIfFailed(reqCtx, backup, request.Backup, err)
+ }
+ request.Status.Actions[i] = mergeActionStatus(&request.Status.Actions[i], status)
- // Failure MetadataCollectionJob does not affect the backup status.
- if err = r.createUpdatesJobs(reqCtx, backup, nil, &snapshotSpec.BasePolicy, backupDestinationPath, dataprotectionv1alpha1.POST); err != nil {
- r.Recorder.Event(backup, corev1.EventTypeNormal, "CreatedPostUpdatesJob", err.Error())
+ switch status.Phase {
+ case dpv1alpha1.ActionPhaseCompleted:
+ updateBackupStatusByActionStatus(&request.Status)
+ continue
+ case dpv1alpha1.ActionPhaseFailed:
+ return r.updateStatusIfFailed(reqCtx, backup, request.Backup,
+ fmt.Errorf("action %s failed, %s", act.GetName(), status.FailureReason))
+ case dpv1alpha1.ActionPhaseRunning:
+ // update status
+ if err = r.Client.Status().Patch(reqCtx.Ctx, request.Backup, client.MergeFrom(backup)); err != nil {
+ return intctrlutil.CheckedRequeueWithError(err, reqCtx.Log, "")
+ }
+ return intctrlutil.Reconciled()
+ }
}
- backup.Status.Phase = dataprotectionv1alpha1.BackupCompleted
- backup.Status.CompletionTimestamp = &metav1.Time{Time: r.clock.Now().UTC()}
- backup.Status.Manifests = &dataprotectionv1alpha1.ManifestsStatus{
- BackupLog: &dataprotectionv1alpha1.BackupLogStatus{
- StartTime: snapshotTime,
- StopTime: snapshotTime,
- },
+ // all actions completed, update backup status to completed
+ request.Status.Phase = dpv1alpha1.BackupPhaseCompleted
+ request.Status.CompletionTimestamp = &metav1.Time{Time: r.clock.Now().UTC()}
+ if !request.Status.StartTimestamp.IsZero() {
+ // round the duration to a multiple of seconds.
+ duration := request.Status.CompletionTimestamp.Sub(request.Status.StartTimestamp.Time).Round(time.Second)
+ request.Status.Duration = &metav1.Duration{Duration: duration}
}
- snap := &snapshotv1.VolumeSnapshot{}
- exists, _ := r.snapshotCli.CheckResourceExists(key, snap)
- if exists {
- backup.Status.TotalSize = snap.Status.RestoreSize.String()
+ r.Recorder.Event(backup, corev1.EventTypeNormal, "CreatedBackup", "Completed backup")
+ if err = r.Client.Status().Patch(reqCtx.Ctx, request.Backup, client.MergeFrom(backup)); err != nil {
+ return intctrlutil.CheckedRequeueWithError(err, reqCtx.Log, "")
}
- return nil, nil
+ return intctrlutil.Reconciled()
}
-// doBaseBackupInProgressPhaseAction handles for base backup during in progress.
-func (r *BackupReconciler) doBaseBackupInProgressPhaseAction(reqCtx intctrlutil.RequestCtx,
- backup *dataprotectionv1alpha1.Backup,
- backupPolicy *dataprotectionv1alpha1.BackupPolicy,
- backupDestinationPath string) (*ctrl.Result, error) {
- // 1. create and ensure backup tool job finished
- // 2. get job phase and update
- commonPolicy := backupPolicy.Spec.GetCommonPolicy(backup.Spec.BackupType)
- if commonPolicy == nil {
- // TODO: add error type
- return intctrlutil.ResultToP(r.updateStatusIfFailed(reqCtx, backup, fmt.Errorf("not found the %s policy", backup.Spec.BackupType)))
- }
- // createUpdatesJobs should not affect the backup status, just need to record events when the run fails
- if err := r.createUpdatesJobs(reqCtx, backup, commonPolicy, &commonPolicy.BasePolicy, backupDestinationPath, dataprotectionv1alpha1.PRE); err != nil {
- r.Recorder.Event(backup, corev1.EventTypeNormal, "CreatedPreUpdatesJob", err.Error())
- }
- if err := r.createBackupToolJob(reqCtx, backup, backupPolicy, commonPolicy, backupDestinationPath); err != nil {
- return intctrlutil.ResultToP(r.updateStatusIfFailed(reqCtx, backup, err))
- }
- key := types.NamespacedName{Namespace: backup.Namespace, Name: backup.Name}
- isOK, err := r.ensureBatchV1JobCompleted(reqCtx, key)
- if err != nil {
- return intctrlutil.ResultToP(r.updateStatusIfFailed(reqCtx, backup, err))
+func mergeActionStatus(original, new *dpv1alpha1.ActionStatus) dpv1alpha1.ActionStatus {
+ as := new.DeepCopy()
+ if original.StartTimestamp != nil {
+ as.StartTimestamp = original.StartTimestamp
}
- if !isOK {
- return intctrlutil.ResultToP(intctrlutil.Reconciled())
- }
- // createUpdatesJobs should not affect the backup status, just need to record events when the run fails
- if err = r.createUpdatesJobs(reqCtx, backup, commonPolicy, &commonPolicy.BasePolicy, backupDestinationPath, dataprotectionv1alpha1.POST); err != nil {
- r.Recorder.Event(backup, corev1.EventTypeNormal, "CreatedPostUpdatesJob", err.Error())
- }
- // updates Phase directly to Completed because `ensureBatchV1JobCompleted` has checked job failed
- backup.Status.Phase = dataprotectionv1alpha1.BackupCompleted
- backup.Status.CompletionTimestamp = &metav1.Time{Time: r.clock.Now().UTC()}
+ return *as
+}
- if backup.Spec.BackupType == dataprotectionv1alpha1.BackupTypeLogFile {
- if backup.Status.Manifests != nil &&
- backup.Status.Manifests.BackupLog != nil &&
- backup.Status.Manifests.BackupLog.StartTime == nil {
- backup.Status.Manifests.BackupLog.StartTime = backup.Status.Manifests.BackupLog.StopTime
+func updateBackupStatusByActionStatus(backupStatus *dpv1alpha1.BackupStatus) {
+ for _, act := range backupStatus.Actions {
+ if act.TotalSize != "" && backupStatus.TotalSize == "" {
+ backupStatus.TotalSize = act.TotalSize
+ }
+ if act.TimeRange != nil && backupStatus.TimeRange == nil {
+ backupStatus.TimeRange = act.TimeRange
}
}
- return nil, nil
}
-func (r *BackupReconciler) doInRunningPhaseAction(
+// handleCompletedPhase handles the backup object in completed phase.
+// It will delete the reference workloads.
+func (r *BackupReconciler) handleCompletedPhase(
reqCtx intctrlutil.RequestCtx,
- backup *dataprotectionv1alpha1.Backup) error {
- backupPolicy, isCompleted, err := r.checkBackupIsCompletedDuringRunning(reqCtx, backup)
- if err != nil {
- return err
- } else if isCompleted {
- return nil
- }
- commonPolicy := backupPolicy.Spec.GetCommonPolicy(backup.Spec.BackupType)
- if commonPolicy == nil {
- return fmt.Errorf(`can not find spec.%s in BackupPolicy "%s"`, strings.ToLower(string(backup.Spec.BackupType)), backupPolicy.Name)
- }
- // reconcile StatefulSet
- sts := &appsv1.StatefulSet{}
- exists, err := intctrlutil.CheckResourceExists(reqCtx.Ctx, r.Client, types.NamespacedName{
- Namespace: backup.Namespace,
- Name: backup.Name,
- }, sts)
- if err != nil {
- return err
- }
- statefulSetSpec, err := r.buildStatefulSpec(reqCtx, backup, backupPolicy, commonPolicy)
- if err != nil {
- return err
- }
- // if not exists, create the statefulSet
- if !exists {
- return r.createBackupStatefulSet(reqCtx, backup, statefulSetSpec)
- }
- sts.Spec.Template = statefulSetSpec.Template
- // update the statefulSet
- if err = r.Update(reqCtx.Ctx, sts); err != nil {
- return err
+ backup *dpv1alpha1.Backup) (ctrl.Result, error) {
+ if err := r.deleteExternalResources(reqCtx, backup); err != nil {
+ return intctrlutil.CheckedRequeueWithError(err, reqCtx.Log, "")
}
- // if available replicas not changed, return
- if backup.Status.AvailableReplicas != nil && *backup.Status.AvailableReplicas == sts.Status.AvailableReplicas {
- return nil
+ return intctrlutil.Reconciled()
+}
+
+func (r *BackupReconciler) updateStatusIfFailed(
+ reqCtx intctrlutil.RequestCtx,
+ original *dpv1alpha1.Backup,
+ backup *dpv1alpha1.Backup,
+ err error) (ctrl.Result, error) {
+ sendWarningEventForError(r.Recorder, backup, err)
+ backup.Status.Phase = dpv1alpha1.BackupPhaseFailed
+ backup.Status.FailureReason = err.Error()
+ if errUpdate := r.Client.Status().Patch(reqCtx.Ctx, backup, client.MergeFrom(original)); errUpdate != nil {
+ return intctrlutil.CheckedRequeueWithError(errUpdate, reqCtx.Log, "")
}
- patch := client.MergeFrom(backup.DeepCopy())
- backup.Status.AvailableReplicas = &sts.Status.AvailableReplicas
- return r.Status().Patch(reqCtx.Ctx, backup, patch)
+ return intctrlutil.CheckedRequeueWithError(err, reqCtx.Log, "")
}
-// checkBackupIsCompletedDuringRunning checks if backup is completed during it is running.
-// it returns ture, if logfile schedule is disabled or cluster is deleted.
-func (r *BackupReconciler) checkBackupIsCompletedDuringRunning(reqCtx intctrlutil.RequestCtx,
- backup *dataprotectionv1alpha1.Backup) (*dataprotectionv1alpha1.BackupPolicy, bool, error) {
- backupPolicy := &dataprotectionv1alpha1.BackupPolicy{}
- exists, err := intctrlutil.CheckResourceExists(reqCtx.Ctx, r.Client, types.NamespacedName{
- Namespace: reqCtx.Req.Namespace,
- Name: backup.Spec.BackupPolicyName,
- }, backupPolicy)
- if err != nil {
- return backupPolicy, false, err
+// deleteExternalJobs deletes the external jobs.
+func (r *BackupReconciler) deleteExternalJobs(reqCtx intctrlutil.RequestCtx, backup *dpv1alpha1.Backup) error {
+ jobs := &batchv1.JobList{}
+ if err := r.Client.List(reqCtx.Ctx, jobs,
+ client.InNamespace(backup.Namespace),
+ client.MatchingLabels(dpbackup.BuildBackupWorkloadLabels(backup))); err != nil {
+ return client.IgnoreNotFound(err)
}
- if exists {
- if err = backup.Spec.Validate(backupPolicy); err != nil {
- return backupPolicy, false, err
+
+ deleteJob := func(job *batchv1.Job) error {
+ if err := dputils.RemoveDataProtectionFinalizer(reqCtx.Ctx, r.Client, job); err != nil {
+ return err
}
- clusterName := backup.Labels[constant.AppInstanceLabelKey]
- targetClusterExists := true
- if clusterName != "" {
- cluster := &appsv1alpha1.Cluster{}
- var err error
- targetClusterExists, err = intctrlutil.CheckResourceExists(reqCtx.Ctx, r.Client, types.NamespacedName{Name: clusterName, Namespace: backup.Namespace}, cluster)
- if err != nil {
- return backupPolicy, false, err
- }
+ if !job.DeletionTimestamp.IsZero() {
+ return nil
}
-
- schedulePolicy := backupPolicy.Spec.GetCommonSchedulePolicy(backup.Spec.BackupType)
- if schedulePolicy.Enable && targetClusterExists {
- return backupPolicy, false, nil
+ reqCtx.Log.V(1).Info("delete job", "job", job)
+ if err := intctrlutil.BackgroundDeleteObject(r.Client, reqCtx.Ctx, job); err != nil {
+ return err
}
+ return nil
}
- patch := client.MergeFrom(backup.DeepCopy())
- backup.Status.Phase = dataprotectionv1alpha1.BackupCompleted
- backup.Status.CompletionTimestamp = &metav1.Time{Time: r.clock.Now().UTC()}
- if !backup.Status.StartTimestamp.IsZero() {
- // round the duration to a multiple of seconds.
- duration := backup.Status.CompletionTimestamp.Sub(backup.Status.StartTimestamp.Time).Round(time.Second)
- backup.Status.Duration = &metav1.Duration{Duration: duration}
+
+ for i := range jobs.Items {
+ if err := deleteJob(&jobs.Items[i]); err != nil {
+ return err
+ }
}
- return backupPolicy, true, r.Client.Status().Patch(reqCtx.Ctx, backup, patch)
+ return nil
}
-func (r *BackupReconciler) createBackupStatefulSet(reqCtx intctrlutil.RequestCtx,
- backup *dataprotectionv1alpha1.Backup,
- stsSpec *appsv1.StatefulSetSpec) error {
- sts := &appsv1.StatefulSet{
- ObjectMeta: metav1.ObjectMeta{
- Name: backup.Name,
- Namespace: backup.Namespace,
- Labels: buildBackupWorkloadsLabels(backup),
- },
- Spec: *stsSpec,
+func (r *BackupReconciler) deleteVolumeSnapshots(reqCtx intctrlutil.RequestCtx,
+ backup *dpv1alpha1.Backup) error {
+ deleter := &dpbackup.Deleter{
+ RequestCtx: reqCtx,
+ Client: r.Client,
}
- controllerutil.AddFinalizer(sts, dataProtectionFinalizerName)
- if err := controllerutil.SetControllerReference(backup, sts, r.Scheme); err != nil {
- return err
- }
- return r.Client.Create(reqCtx.Ctx, sts)
+ return deleter.DeleteVolumeSnapshots(backup)
}
-func (r *BackupReconciler) buildManifestsUpdaterContainer(backup *dataprotectionv1alpha1.Backup,
- commonPolicy *dataprotectionv1alpha1.CommonBackupPolicy,
- backupDestinationPath string) (corev1.Container, error) {
- container := corev1.Container{}
- cueFS, _ := debme.FS(cueTemplates, "cue")
- cueTpl, err := intctrlutil.NewCUETplFromBytes(cueFS.ReadFile("manifests_updater.cue"))
- if err != nil {
- return container, err
- }
- cueValue := intctrlutil.NewCUEBuilder(*cueTpl)
- optionsBytes, err := json.Marshal(map[string]string{
- "backupName": backup.Name,
- "namespace": backup.Namespace,
- "image": viper.GetString(constant.KBToolsImage),
- "containerName": manifestsUpdaterContainerName,
- "imagePullPolicy": viper.GetString(constant.KBImagePullPolicy),
- })
- if err != nil {
- return container, err
- }
- if err = cueValue.Fill("options", optionsBytes); err != nil {
- return container, err
- }
- containerBytes, err := cueValue.Lookup("container")
- if err != nil {
- return container, err
+// deleteExternalStatefulSet deletes the external statefulSet.
+func (r *BackupReconciler) deleteExternalStatefulSet(reqCtx intctrlutil.RequestCtx, backup *dpv1alpha1.Backup) error {
+ key := client.ObjectKey{
+ Namespace: backup.Namespace,
+ Name: backup.Name,
}
- if err = json.Unmarshal(containerBytes, &container); err != nil {
- return container, err
+ sts := &appsv1.StatefulSet{}
+ if err := r.Client.Get(reqCtx.Ctx, key, sts); err != nil {
+ return client.IgnoreNotFound(err)
+ } else if !model.IsOwnerOf(backup, sts) {
+ return nil
}
- container.VolumeMounts = []corev1.VolumeMount{
- {Name: fmt.Sprintf("backup-%s", backup.Status.PersistentVolumeClaimName), MountPath: backupPathBase},
+
+ patch := client.MergeFrom(sts.DeepCopy())
+ controllerutil.RemoveFinalizer(sts, dptypes.DataProtectionFinalizerName)
+ if err := r.Client.Patch(reqCtx.Ctx, sts, patch); err != nil {
+ return err
}
- intctrlutil.InjectZeroResourcesLimitsIfEmpty(&container)
- container.Env = []corev1.EnvVar{
- {Name: constant.DPBackupInfoFile, Value: buildBackupInfoENV(backupDestinationPath)},
+
+ if !sts.DeletionTimestamp.IsZero() {
+ return nil
}
- return container, nil
+
+ reqCtx.Log.V(1).Info("delete statefulSet", "statefulSet", sts)
+ return intctrlutil.BackgroundDeleteObject(r.Client, reqCtx.Ctx, sts)
}
-func (r *BackupReconciler) buildStatefulSpec(reqCtx intctrlutil.RequestCtx,
- backup *dataprotectionv1alpha1.Backup,
- backupPolicy *dataprotectionv1alpha1.BackupPolicy,
- commonPolicy *dataprotectionv1alpha1.CommonBackupPolicy) (*appsv1.StatefulSetSpec, error) {
- backupDestinationPath := getBackupDestinationPath(backup, backupPolicy.Annotations[constant.BackupDataPathPrefixAnnotationKey])
- toolPodSpec, err := r.buildBackupToolPodSpec(reqCtx, backup, backupPolicy, commonPolicy, backupDestinationPath)
- toolPodSpec.RestartPolicy = corev1.RestartPolicyAlways
- if err != nil {
- return nil, err
- }
- // build the manifests updater container for backup.status.manifests
- manifestsUpdaterContainer, err := r.buildManifestsUpdaterContainer(backup, commonPolicy, backupDestinationPath)
- if err != nil {
- return nil, err
- }
- // build ARCHIVE_INTERVAL env
- schedulePolicy := backupPolicy.Spec.GetCommonSchedulePolicy(backup.Spec.BackupType)
- interval := getIntervalSecondsForLogfile(backup.Spec.BackupType, schedulePolicy.CronExpression)
- if interval != "" {
- toolPodSpec.Containers[0].Env = append(toolPodSpec.Containers[0].Env, corev1.EnvVar{
- Name: constant.DPArchiveInterval,
- Value: interval,
- })
- }
- target, _ := r.getTargetPod(reqCtx, backup, commonPolicy.Target.LabelsSelector.MatchLabels)
- if target != nil && target.Spec.ServiceAccountName != "" {
- toolPodSpec.Containers = append(toolPodSpec.Containers, manifestsUpdaterContainer)
- toolPodSpec.ServiceAccountName = target.Spec.ServiceAccountName
- }
- backupLabels := buildBackupWorkloadsLabels(backup)
- defaultReplicas := int32(1)
- return &appsv1.StatefulSetSpec{
- Replicas: &defaultReplicas,
- Selector: &metav1.LabelSelector{
- MatchLabels: backupLabels,
- },
- Template: corev1.PodTemplateSpec{
- ObjectMeta: metav1.ObjectMeta{
- Labels: backupLabels,
- },
- Spec: toolPodSpec,
- },
- }, nil
-}
-
-func (r *BackupReconciler) doCompletedPhaseAction(
- reqCtx intctrlutil.RequestCtx,
- backup *dataprotectionv1alpha1.Backup) (ctrl.Result, error) {
-
- if err := r.deleteReferenceBatchV1Jobs(reqCtx, backup); err != nil && !apierrors.IsNotFound(err) {
- return intctrlutil.CheckedRequeueWithError(err, reqCtx.Log, "")
- }
-
- if err := r.deleteReferenceStatefulSet(reqCtx, backup); err != nil && !apierrors.IsNotFound(err) {
- return intctrlutil.CheckedRequeueWithError(err, reqCtx.Log, "")
- }
- return intctrlutil.Reconciled()
-}
-
-func (r *BackupReconciler) updateStatusIfFailed(reqCtx intctrlutil.RequestCtx,
- backup *dataprotectionv1alpha1.Backup, err error) (ctrl.Result, error) {
- patch := client.MergeFrom(backup.DeepCopy())
- sendWarningEventForError(r.Recorder, backup, err)
- backup.Status.Phase = dataprotectionv1alpha1.BackupFailed
- backup.Status.FailureReason = err.Error()
- if errUpdate := r.Client.Status().Patch(reqCtx.Ctx, backup, patch); errUpdate != nil {
- return intctrlutil.CheckedRequeueWithError(errUpdate, reqCtx.Log, "")
- }
- return intctrlutil.CheckedRequeueWithError(err, reqCtx.Log, "")
-}
-
-// getCluster gets the cluster and will ignore the error.
-func (r *BackupReconciler) getCluster(
- reqCtx intctrlutil.RequestCtx,
- targetPod *corev1.Pod) *appsv1alpha1.Cluster {
- clusterName := targetPod.Labels[constant.AppInstanceLabelKey]
- if len(clusterName) == 0 {
- return nil
- }
- cluster := &appsv1alpha1.Cluster{}
- if err := r.Client.Get(reqCtx.Ctx, types.NamespacedName{
- Namespace: targetPod.Namespace,
- Name: clusterName,
- }, cluster); err != nil {
- // should not affect the backup status
- return nil
- }
- return cluster
-}
-
-// patchBackupObjectLabels add missed labels to the backup object.
-func (r *BackupReconciler) patchBackupObjectLabels(
- reqCtx intctrlutil.RequestCtx,
- backup *dataprotectionv1alpha1.Backup,
- labels map[string]string) (bool, error) {
- oldBackup := backup.DeepCopy()
- if backup.Labels == nil {
- backup.Labels = make(map[string]string)
- }
- for k, v := range labels {
- backup.Labels[k] = v
- }
- if reflect.DeepEqual(oldBackup.ObjectMeta, backup.ObjectMeta) {
- return false, nil
- }
- return true, r.Client.Patch(reqCtx.Ctx, backup, client.MergeFrom(oldBackup))
-}
-
-// patchBackupObjectMeta patches backup object metaObject include cluster snapshot.
-func (r *BackupReconciler) patchBackupObjectMeta(
- reqCtx intctrlutil.RequestCtx,
- backup *dataprotectionv1alpha1.Backup,
- targetPod *corev1.Pod,
- cluster *appsv1alpha1.Cluster,
- updateLabels map[string]string) (bool, error) {
- if backup.Labels == nil {
- backup.Labels = make(map[string]string)
- }
- oldBackup := backup.DeepCopy()
- if cluster != nil {
- if err := r.setClusterSnapshotAnnotation(backup, cluster); err != nil {
- return false, err
- }
- backup.Labels[constant.DataProtectionLabelClusterUIDKey] = string(cluster.UID)
- }
- for _, v := range getClusterLabelKeys() {
- backup.Labels[v] = targetPod.Labels[v]
- }
- backup.Labels[constant.AppManagedByLabelKey] = constant.AppName
- backup.Labels[dataProtectionLabelBackupTypeKey] = string(backup.Spec.BackupType)
- for k, v := range updateLabels {
- backup.Labels[k] = v
- }
- if backup.Annotations == nil {
- backup.Annotations = make(map[string]string)
- }
- backup.Annotations[dataProtectionBackupTargetPodKey] = targetPod.Name
- controllerutil.AddFinalizer(backup, dataProtectionFinalizerName)
- if reflect.DeepEqual(oldBackup.ObjectMeta, backup.ObjectMeta) {
- return false, nil
- }
- return true, r.Client.Patch(reqCtx.Ctx, backup, client.MergeFrom(oldBackup))
-}
-
-func (r *BackupReconciler) createPreCommandJobAndEnsure(reqCtx intctrlutil.RequestCtx,
- backup *dataprotectionv1alpha1.Backup,
- snapshotPolicy *dataprotectionv1alpha1.SnapshotPolicy) (bool, error) {
-
- emptyCmd, err := r.ensureEmptyHooksCommand(snapshotPolicy, true)
- if err != nil {
- return false, err
- }
- // if undefined commands, skip create job.
- if emptyCmd {
- return true, err
- }
-
- mgrNS := viper.GetString(constant.CfgKeyCtrlrMgrNS)
- key := types.NamespacedName{Namespace: mgrNS, Name: generateUniqueJobName(backup, "hook-pre")}
- if err := r.createHooksCommandJob(reqCtx, backup, snapshotPolicy, key, true); err != nil {
- return false, err
- }
- return r.ensureBatchV1JobCompleted(reqCtx, key)
-}
-
-func (r *BackupReconciler) createPostCommandJobAndEnsure(reqCtx intctrlutil.RequestCtx,
- backup *dataprotectionv1alpha1.Backup,
- snapshotPolicy *dataprotectionv1alpha1.SnapshotPolicy) (bool, error) {
-
- emptyCmd, err := r.ensureEmptyHooksCommand(snapshotPolicy, false)
- if err != nil {
- return false, err
- }
- // if undefined commands, skip create job.
- if emptyCmd {
- return true, err
- }
-
- mgrNS := viper.GetString(constant.CfgKeyCtrlrMgrNS)
- key := types.NamespacedName{Namespace: mgrNS, Name: generateUniqueJobName(backup, "hook-post")}
- if err = r.createHooksCommandJob(reqCtx, backup, snapshotPolicy, key, false); err != nil {
- return false, err
- }
- return r.ensureBatchV1JobCompleted(reqCtx, key)
-}
-
-func (r *BackupReconciler) ensureBatchV1JobCompleted(
- reqCtx intctrlutil.RequestCtx, key types.NamespacedName) (bool, error) {
- job := &batchv1.Job{}
- exists, err := intctrlutil.CheckResourceExists(reqCtx.Ctx, r.Client, key, job)
- if err != nil {
- return false, err
- }
- if exists {
- if containsJobCondition(job, batchv1.JobComplete) {
- return true, nil
- }
- if containsJobCondition(job, batchv1.JobFailed) {
- return false, intctrlutil.NewBackupJobFailed(job.Name)
- }
- }
- return false, nil
-}
-
-func (r *BackupReconciler) createVolumeSnapshot(
- reqCtx intctrlutil.RequestCtx,
- backup *dataprotectionv1alpha1.Backup,
- snapshotPolicy *dataprotectionv1alpha1.SnapshotPolicy) error {
-
- snap := &snapshotv1.VolumeSnapshot{}
- exists, err := r.snapshotCli.CheckResourceExists(reqCtx.Req.NamespacedName, snap)
- if err != nil {
- return err
- }
- if exists {
- // find resource object, skip created.
- return nil
- }
-
- // get backup policy
- backupPolicy := &dataprotectionv1alpha1.BackupPolicy{}
- backupPolicyNameSpaceName := types.NamespacedName{
- Namespace: reqCtx.Req.Namespace,
- Name: backup.Spec.BackupPolicyName,
- }
-
- if err := r.Get(reqCtx.Ctx, backupPolicyNameSpaceName, backupPolicy); err != nil {
- reqCtx.Log.Error(err, "Unable to get backupPolicy for backup.", "backupPolicy", backupPolicyNameSpaceName)
- return err
- }
-
- targetPVCs, err := r.getTargetPVCs(reqCtx, backup, snapshotPolicy.Target.LabelsSelector.MatchLabels)
- if err != nil {
- return err
- }
- for _, target := range targetPVCs {
- snapshotName := backup.Name
- vsc := snapshotv1.VolumeSnapshotClass{}
- if target.Spec.StorageClassName != nil {
- if err = r.getVolumeSnapshotClassOrCreate(reqCtx.Ctx, *target.Spec.StorageClassName, &vsc); err != nil {
- return err
- }
- }
- labels := buildBackupWorkloadsLabels(backup)
- labels[constant.VolumeTypeLabelKey] = target.Labels[constant.VolumeTypeLabelKey]
- if target.Labels[constant.VolumeTypeLabelKey] == string(appsv1alpha1.VolumeTypeLog) {
- snapshotName += "-log"
- }
- snap = &snapshotv1.VolumeSnapshot{
- ObjectMeta: metav1.ObjectMeta{
- Namespace: reqCtx.Req.Namespace,
- Name: snapshotName,
- Labels: labels,
- },
- Spec: snapshotv1.VolumeSnapshotSpec{
- Source: snapshotv1.VolumeSnapshotSource{
- PersistentVolumeClaimName: &target.Name,
- },
- VolumeSnapshotClassName: &vsc.Name,
- },
- }
-
- controllerutil.AddFinalizer(snap, dataProtectionFinalizerName)
- if err = controllerutil.SetControllerReference(backup, snap, r.Scheme); err != nil {
- return err
- }
-
- reqCtx.Log.V(1).Info("create a volumeSnapshot from backup", "snapshot", snap.Name)
- if err = r.snapshotCli.Create(snap); err != nil && !apierrors.IsAlreadyExists(err) {
- return err
- }
- }
- msg := fmt.Sprintf("Waiting for the volume snapshot %s creation to complete in backup.", snap.Name)
- r.Recorder.Event(backup, corev1.EventTypeNormal, "CreatingVolumeSnapshot", msg)
- return nil
-}
-
-func (r *BackupReconciler) getVolumeSnapshotClassOrCreate(ctx context.Context, storageClassName string, vsc *snapshotv1.VolumeSnapshotClass) error {
- storageClassObj := storagev1.StorageClass{}
- if err := r.Client.Get(ctx, types.NamespacedName{Name: storageClassName}, &storageClassObj); err != nil {
- // ignore if not found storage class, use the default volume snapshot class
- return client.IgnoreNotFound(err)
- }
- vscList := snapshotv1.VolumeSnapshotClassList{}
- if err := r.snapshotCli.List(&vscList); err != nil {
- return err
- }
- for _, item := range vscList.Items {
- if item.Driver == storageClassObj.Provisioner {
- *vsc = item
- return nil
- }
- }
- // not found matched volume snapshot class, create one
- vscName := fmt.Sprintf("vsc-%s-%s", storageClassName, storageClassObj.UID[:8])
- newVSC, err := ctrlbuilder.BuildVolumeSnapshotClass(vscName, storageClassObj.Provisioner)
- if err != nil {
- return err
- }
- if err = r.snapshotCli.Create(newVSC); err != nil {
- return err
- }
- *vsc = *newVSC
- return nil
-}
-
-func (r *BackupReconciler) ensureVolumeSnapshotReady(
- key types.NamespacedName) (bool, *metav1.Time, error) {
- snap := &snapshotv1.VolumeSnapshot{}
- // not found, continue the creation process
- exists, err := r.snapshotCli.CheckResourceExists(key, snap)
- if err != nil {
- return false, nil, err
- }
- if exists && snap.Status != nil {
- // check if snapshot status throws an error, e.g. csi does not support volume snapshot
- if isVolumeSnapshotConfigError(snap) {
- return false, nil, errors.New(*snap.Status.Error.Message)
- }
- if snap.Status.ReadyToUse != nil && *snap.Status.ReadyToUse {
- return true, snap.Status.CreationTime, nil
- }
- }
- return false, nil, nil
-}
-
-func (r *BackupReconciler) createUpdatesJobs(reqCtx intctrlutil.RequestCtx,
- backup *dataprotectionv1alpha1.Backup,
- commonPolicy *dataprotectionv1alpha1.CommonBackupPolicy,
- basePolicy *dataprotectionv1alpha1.BasePolicy,
- backupDestinationPath string,
- stage dataprotectionv1alpha1.BackupStatusUpdateStage) error {
- // get backup policy
- backupPolicy := &dataprotectionv1alpha1.BackupPolicy{}
- backupPolicyNameSpaceName := types.NamespacedName{
- Namespace: reqCtx.Req.Namespace,
- Name: backup.Spec.BackupPolicyName,
- }
- if err := r.Get(reqCtx.Ctx, backupPolicyNameSpaceName, backupPolicy); err != nil {
- reqCtx.Log.V(1).Error(err, "Unable to get backupPolicy for backup.", "backupPolicy", backupPolicyNameSpaceName)
- return err
- }
- for index, update := range basePolicy.BackupStatusUpdates {
- if update.UpdateStage != stage {
- continue
- }
- if err := r.createMetadataCollectionJob(reqCtx, backup, commonPolicy, basePolicy, backupDestinationPath, update, index); err != nil {
- return err
- }
- }
- return nil
-}
-
-func (r *BackupReconciler) createMetadataCollectionJob(reqCtx intctrlutil.RequestCtx,
- backup *dataprotectionv1alpha1.Backup,
- commonPolicy *dataprotectionv1alpha1.CommonBackupPolicy,
- basePolicy *dataprotectionv1alpha1.BasePolicy,
- backupDestinationPath string,
- updateInfo dataprotectionv1alpha1.BackupStatusUpdate,
- index int) error {
- jobNamespace := viper.GetString(constant.CfgKeyCtrlrMgrNS)
- // if specified to use the service account of target pod, the namespace should be the namespace of backup.
- if updateInfo.UseTargetPodServiceAccount {
- jobNamespace = backup.Namespace
- }
- key := types.NamespacedName{Namespace: jobNamespace, Name: generateUniqueJobName(backup, fmt.Sprintf("status-%d-%s", index, string(updateInfo.UpdateStage)))}
- job := &batchv1.Job{}
- // check if job is created
- if exists, err := intctrlutil.CheckResourceExists(reqCtx.Ctx, r.Client, key, job); err != nil {
- return err
- } else if exists {
- return nil
- }
-
- // build job and create
- jobPodSpec, err := r.buildMetadataCollectionPodSpec(reqCtx, backup, commonPolicy, basePolicy, backupDestinationPath, updateInfo)
- if err != nil {
- return err
- }
- if job, err = ctrlbuilder.BuildBackupManifestsJob(key, backup, &jobPodSpec); err != nil {
- return err
- }
- msg := fmt.Sprintf("creating job %s", key.Name)
- r.Recorder.Event(backup, corev1.EventTypeNormal, "CreatingJob-"+key.Name, msg)
- return client.IgnoreAlreadyExists(r.Client.Create(reqCtx.Ctx, job))
-}
-
-func (r *BackupReconciler) createDeleteBackupFileJob(
- reqCtx intctrlutil.RequestCtx,
- jobKey types.NamespacedName,
- backup *dataprotectionv1alpha1.Backup,
- backupPVCName string,
- backupFilePath string) error {
-
- // make sure the path has a leading slash
- if !strings.HasPrefix(backupFilePath, "/") {
- backupFilePath = "/" + backupFilePath
- }
-
- // this script first deletes the directory where the backup is located (including files
- // in the directory), and then traverses up the path level by level to clean up empty directories.
- deleteScript := fmt.Sprintf(`
- backupPathBase=%s;
- targetPath="${backupPathBase}%s";
-
- echo "removing backup files in ${targetPath}";
- rm -rf "${targetPath}";
-
- absBackupPathBase=$(realpath "${backupPathBase}");
- curr=$(realpath "${targetPath}");
- while true; do
- parent=$(dirname "${curr}");
- if [ "${parent}" == "${absBackupPathBase}" ]; then
- echo "reach backupPathBase ${backupPathBase}, done";
- break;
- fi;
- if [ ! "$(ls -A "${parent}")" ]; then
- echo "${parent} is empty, removing it...";
- rmdir "${parent}";
- else
- echo "${parent} is not empty, done";
- break;
- fi;
- curr="${parent}";
- done
- `, backupPathBase, backupFilePath)
-
- // build container
- container := corev1.Container{}
- container.Name = backup.Name
- container.Command = []string{"sh", "-c"}
- container.Args = []string{deleteScript}
- container.Image = viper.GetString(constant.KBToolsImage)
- container.ImagePullPolicy = corev1.PullPolicy(viper.GetString(constant.KBImagePullPolicy))
-
- allowPrivilegeEscalation := false
- runAsUser := int64(0)
- container.SecurityContext = &corev1.SecurityContext{
- AllowPrivilegeEscalation: &allowPrivilegeEscalation,
- RunAsUser: &runAsUser,
- }
- intctrlutil.InjectZeroResourcesLimitsIfEmpty(&container)
-
- // build pod
- podSpec := corev1.PodSpec{
- Containers: []corev1.Container{container},
- RestartPolicy: corev1.RestartPolicyNever,
- }
-
- // mount the backup volume to the pod
- r.appendBackupVolumeMount(backupPVCName, &podSpec, &podSpec.Containers[0])
-
- if err := addTolerations(&podSpec); err != nil {
- return err
- }
-
- // build job
- backOffLimit := int32(3)
- job := &batchv1.Job{
- ObjectMeta: metav1.ObjectMeta{
- Namespace: jobKey.Namespace,
- Name: jobKey.Name,
- },
- Spec: batchv1.JobSpec{
- Template: corev1.PodTemplateSpec{
- ObjectMeta: metav1.ObjectMeta{
- Namespace: jobKey.Namespace,
- Name: jobKey.Name,
- },
- Spec: podSpec,
- },
- BackoffLimit: &backOffLimit,
- },
- }
- if err := controllerutil.SetControllerReference(backup, job, r.Scheme); err != nil {
- return err
- }
- reqCtx.Log.V(1).Info("create a job from delete backup files", "job", job)
- return client.IgnoreAlreadyExists(r.Client.Create(reqCtx.Ctx, job))
-}
-
-func (r *BackupReconciler) createBackupToolJob(
- reqCtx intctrlutil.RequestCtx,
- backup *dataprotectionv1alpha1.Backup,
- backupPolicy *dataprotectionv1alpha1.BackupPolicy,
- commonPolicy *dataprotectionv1alpha1.CommonBackupPolicy,
- backupDestinationPath string) error {
-
- key := types.NamespacedName{Namespace: backup.Namespace, Name: backup.Name}
- job := batchv1.Job{}
- exists, err := intctrlutil.CheckResourceExists(reqCtx.Ctx, r.Client, key, &job)
- if err != nil {
- return err
- }
- if exists {
- // find resource object, skip created.
- return nil
- }
-
- toolPodSpec, err := r.buildBackupToolPodSpec(reqCtx, backup, backupPolicy, commonPolicy, backupDestinationPath)
- if err != nil {
- return err
- }
-
- if err = r.createBatchV1Job(reqCtx, key, backup, toolPodSpec); err != nil {
- return err
- }
- msg := fmt.Sprintf("Waiting for the job %s creation to complete.", key.Name)
- r.Recorder.Event(backup, corev1.EventTypeNormal, "CreatingJob", msg)
- return nil
-}
-
-// ensureEmptyHooksCommand determines whether it has empty commands in the hooks
-func (r *BackupReconciler) ensureEmptyHooksCommand(
- snapshotPolicy *dataprotectionv1alpha1.SnapshotPolicy,
- preCommand bool) (bool, error) {
- // return true directly, means hooks commands is empty, skip subsequent hook jobs.
- if snapshotPolicy.Hooks == nil {
- return true, nil
- }
-
- commands := snapshotPolicy.Hooks.PostCommands
- if preCommand {
- commands = snapshotPolicy.Hooks.PreCommands
- }
- if len(commands) == 0 {
- return true, nil
- }
- return false, nil
-}
-
-func (r *BackupReconciler) createHooksCommandJob(
- reqCtx intctrlutil.RequestCtx,
- backup *dataprotectionv1alpha1.Backup,
- snapshotPolicy *dataprotectionv1alpha1.SnapshotPolicy,
- key types.NamespacedName,
- preCommand bool) error {
-
- job := batchv1.Job{}
- exists, err := intctrlutil.CheckResourceExists(reqCtx.Ctx, r.Client, key, &job)
- if err != nil {
- return err
- }
- if exists {
- // find resource object, skip created.
- return nil
- }
-
- jobPodSpec, err := r.buildSnapshotPodSpec(reqCtx, backup, snapshotPolicy, preCommand)
- if err != nil {
- return err
- }
-
- msg := fmt.Sprintf("Waiting for the job %s creation to complete.", key.Name)
- r.Recorder.Event(backup, corev1.EventTypeNormal, "CreatingJob-"+key.Name, msg)
-
- return r.createBatchV1Job(reqCtx, key, backup, jobPodSpec)
-}
-
-func (r *BackupReconciler) createBatchV1Job(
- reqCtx intctrlutil.RequestCtx,
- key types.NamespacedName,
- backup *dataprotectionv1alpha1.Backup,
- templatePodSpec corev1.PodSpec) error {
-
- backOffLimit := int32(3)
- job := &batchv1.Job{
- // TypeMeta: metav1.TypeMeta{Kind: "Job", APIVersion: "batch/v1"},
- ObjectMeta: metav1.ObjectMeta{
- Namespace: key.Namespace,
- Name: key.Name,
- Labels: buildBackupWorkloadsLabels(backup),
- },
- Spec: batchv1.JobSpec{
- Template: corev1.PodTemplateSpec{
- ObjectMeta: metav1.ObjectMeta{
- Namespace: key.Namespace,
- Name: key.Name},
- Spec: templatePodSpec,
- },
- BackoffLimit: &backOffLimit,
- },
- }
- controllerutil.AddFinalizer(job, dataProtectionFinalizerName)
- if backup.Namespace == job.Namespace {
- if err := controllerutil.SetControllerReference(backup, job, r.Scheme); err != nil {
- return err
- }
- }
-
- reqCtx.Log.V(1).Info("create a built-in job from backup", "job", job)
- return client.IgnoreAlreadyExists(r.Client.Create(reqCtx.Ctx, job))
-}
-
-func (r *BackupReconciler) deleteReferenceBatchV1Jobs(reqCtx intctrlutil.RequestCtx, backup *dataprotectionv1alpha1.Backup) error {
- jobs := &batchv1.JobList{}
- namespace := backup.Namespace
- if backup.Spec.BackupType == dataprotectionv1alpha1.BackupTypeSnapshot {
- namespace = viper.GetString(constant.CfgKeyCtrlrMgrNS)
- }
- if err := r.Client.List(reqCtx.Ctx, jobs,
- client.InNamespace(namespace),
- client.MatchingLabels(buildBackupWorkloadsLabels(backup))); err != nil {
- return err
- }
-
- for _, job := range jobs.Items {
- if controllerutil.ContainsFinalizer(&job, dataProtectionFinalizerName) {
- patch := client.MergeFrom(job.DeepCopy())
- controllerutil.RemoveFinalizer(&job, dataProtectionFinalizerName)
- if err := r.Patch(reqCtx.Ctx, &job, patch); err != nil {
- return err
- }
- }
-
- if err := intctrlutil.BackgroundDeleteObject(r.Client, reqCtx.Ctx, &job); err != nil {
- return err
- }
- }
- return nil
-}
-
-func (r *BackupReconciler) deleteReferenceVolumeSnapshot(reqCtx intctrlutil.RequestCtx, backup *dataprotectionv1alpha1.Backup) error {
- snaps := &snapshotv1.VolumeSnapshotList{}
-
- if err := r.snapshotCli.List(snaps,
- client.InNamespace(reqCtx.Req.Namespace),
- client.MatchingLabels(buildBackupWorkloadsLabels(backup))); err != nil {
- return err
- }
- for _, i := range snaps.Items {
- if controllerutil.ContainsFinalizer(&i, dataProtectionFinalizerName) {
- patch := i.DeepCopy()
- controllerutil.RemoveFinalizer(&i, dataProtectionFinalizerName)
- if err := r.snapshotCli.Patch(&i, patch); err != nil {
- return err
- }
- }
- if err := r.snapshotCli.Delete(&i); err != nil {
- return err
- }
- }
- return nil
-}
-
-func (r *BackupReconciler) handleDeleteBackupFiles(reqCtx intctrlutil.RequestCtx, backup *dataprotectionv1alpha1.Backup) (*batchv1.Job, error) {
- if backup.Spec.BackupType == dataprotectionv1alpha1.BackupTypeSnapshot {
- // no file to delete for this type
- return nil, nil
- }
- if backup.Status.Phase == dataprotectionv1alpha1.BackupNew {
- // nothing to delete
- return nil, nil
- }
- jobKey := buildDeleteBackupFilesJobNamespacedName(backup)
- job := &batchv1.Job{}
- exists, err := intctrlutil.CheckResourceExists(reqCtx.Ctx, r.Client, jobKey, job)
- if err != nil {
- return nil, err
- }
- // create job for deleting backup files
- if !exists {
- pvcName := backup.Status.PersistentVolumeClaimName
- if pvcName == "" {
- reqCtx.Log.Info("skip deleting backup files because PersistentVolumeClaimName is empty",
- "backup", backup.Name)
- return nil, nil
- }
- // check if pvc exists
- if err = r.Client.Get(reqCtx.Ctx, types.NamespacedName{Namespace: backup.Namespace, Name: pvcName}, &corev1.PersistentVolumeClaim{}); err != nil {
- if apierrors.IsNotFound(err) {
- return nil, nil
- }
- return nil, err
- }
-
- backupFilePath := ""
- if backup.Status.Manifests != nil && backup.Status.Manifests.BackupTool != nil {
- backupFilePath = backup.Status.Manifests.BackupTool.FilePath
- }
- if backupFilePath == "" || !strings.Contains(backupFilePath, backup.Name) {
- // For compatibility: the FilePath field is changing from time to time,
- // and it may not contain the backup name as a path component if the Backup object
- // was created in a previous version. In this case, it's dangerous to execute
- // the deletion command. For example, files belongs to other Backups can be deleted as well.
- reqCtx.Log.Info("skip deleting backup files because backupFilePath is invalid",
- "backupFilePath", backupFilePath, "backup", backup.Name)
- return nil, nil
- }
- // the job will run in the background
- return job, r.createDeleteBackupFileJob(reqCtx, jobKey, backup, pvcName, backupFilePath)
- }
- return job, nil
-}
-
-// deleteReferenceStatefulSet deletes the referenced statefulSet.
-func (r *BackupReconciler) deleteReferenceStatefulSet(reqCtx intctrlutil.RequestCtx, backup *dataprotectionv1alpha1.Backup) error {
- sts := &appsv1.StatefulSet{}
- exists, err := intctrlutil.CheckResourceExists(reqCtx.Ctx, r.Client, types.NamespacedName{
- Namespace: backup.Namespace,
- Name: backup.Name,
- }, sts)
- if err != nil {
+// deleteExternalResources deletes the external workloads that execute backup.
+// Currently, it only supports two types of workloads: statefulSet and job.
+func (r *BackupReconciler) deleteExternalResources(
+ reqCtx intctrlutil.RequestCtx, backup *dpv1alpha1.Backup) error {
+ if err := r.deleteExternalStatefulSet(reqCtx, backup); err != nil {
return err
}
- if !exists && !model.IsOwnerOf(backup, sts) {
- return nil
- }
- patch := client.MergeFrom(sts.DeepCopy())
- controllerutil.RemoveFinalizer(sts, dataProtectionFinalizerName)
- if err = r.Client.Patch(reqCtx.Ctx, sts, patch); err != nil {
- return err
- }
- return intctrlutil.BackgroundDeleteObject(r.Client, reqCtx.Ctx, sts)
-}
-
-func (r *BackupReconciler) deleteExternalResources(reqCtx intctrlutil.RequestCtx, backup *dataprotectionv1alpha1.Backup) error {
- if err := r.deleteReferenceStatefulSet(reqCtx, backup); err != nil {
- return err
- }
- if err := r.deleteReferenceBatchV1Jobs(reqCtx, backup); err != nil {
- return err
- }
- if err := r.deleteReferenceVolumeSnapshot(reqCtx, backup); err != nil {
- return err
- }
- return nil
-}
-
-// getTargetPod gets the target pod by label selector.
-// if the backup has obtained the target pod from label selector, it will be set to the annotations.
-// then get the pod from this annotation to ensure that the same pod is picked up in future.
-func (r *BackupReconciler) getTargetPod(reqCtx intctrlutil.RequestCtx,
- backup *dataprotectionv1alpha1.Backup, labels map[string]string) (*corev1.Pod, error) {
- reqCtx.Log.V(1).Info("Get pod from label", "label", labels)
- targetPod := &corev1.PodList{}
- if err := r.Client.List(reqCtx.Ctx, targetPod,
- client.InNamespace(reqCtx.Req.Namespace),
- client.MatchingLabels(labels)); err != nil {
- return nil, err
- }
- if len(targetPod.Items) == 0 {
- return nil, errors.New("can not find any pod to backup by labelsSelector")
- }
- sort.Sort(intctrlutil.ByPodName(targetPod.Items))
- targetPodName := backup.Annotations[dataProtectionBackupTargetPodKey]
- for _, v := range targetPod.Items {
- if targetPodName == v.Name {
- return &v, nil
- }
- }
- return &targetPod.Items[0], nil
-}
-
-func (r *BackupReconciler) getTargetPVCs(reqCtx intctrlutil.RequestCtx,
- backup *dataprotectionv1alpha1.Backup, podLabels map[string]string) ([]corev1.PersistentVolumeClaim, error) {
- targetPod, err := r.getTargetPod(reqCtx, backup, podLabels)
- if err != nil {
- return nil, err
- }
- tempPVC := corev1.PersistentVolumeClaim{}
- var dataPVC *corev1.PersistentVolumeClaim
- var logPVC *corev1.PersistentVolumeClaim
- for _, volume := range targetPod.Spec.Volumes {
- if volume.PersistentVolumeClaim == nil {
- continue
- }
- pvcKey := types.NamespacedName{Namespace: backup.Namespace, Name: volume.PersistentVolumeClaim.ClaimName}
- if err = r.Client.Get(reqCtx.Ctx, pvcKey, &tempPVC); err != nil && !apierrors.IsNotFound(err) {
- return nil, err
- }
- switch tempPVC.Labels[constant.VolumeTypeLabelKey] {
- case string(appsv1alpha1.VolumeTypeData):
- dataPVC = tempPVC.DeepCopy()
- case string(appsv1alpha1.VolumeTypeLog):
- logPVC = tempPVC.DeepCopy()
- }
- }
-
- if dataPVC == nil {
- return nil, errors.New("can not find any pvc to backup with labelsSelector")
- }
-
- allPVCs := []corev1.PersistentVolumeClaim{*dataPVC}
- if logPVC != nil {
- allPVCs = append(allPVCs, *logPVC)
- }
-
- return allPVCs, nil
-}
-
-func (r *BackupReconciler) appendBackupVolumeMount(
- pvcName string,
- podSpec *corev1.PodSpec,
- container *corev1.Container) {
- // TODO(dsj): mount multi remote backup volumes
- remoteVolumeName := fmt.Sprintf("backup-%s", pvcName)
- remoteVolume := corev1.Volume{
- Name: remoteVolumeName,
- VolumeSource: corev1.VolumeSource{
- PersistentVolumeClaim: &corev1.PersistentVolumeClaimVolumeSource{
- ClaimName: pvcName,
- },
- },
- }
- remoteVolumeMount := corev1.VolumeMount{
- Name: remoteVolumeName,
- MountPath: backupPathBase,
- }
- podSpec.Volumes = append(podSpec.Volumes, remoteVolume)
- container.VolumeMounts = append(container.VolumeMounts, remoteVolumeMount)
-}
-
-func (r *BackupReconciler) buildBackupToolPodSpec(reqCtx intctrlutil.RequestCtx,
- backup *dataprotectionv1alpha1.Backup,
- backupPolicy *dataprotectionv1alpha1.BackupPolicy,
- commonPolicy *dataprotectionv1alpha1.CommonBackupPolicy,
- pathPrefix string) (corev1.PodSpec, error) {
- podSpec := corev1.PodSpec{}
- // get backup tool
- backupTool, err := getBackupToolByName(reqCtx, r.Client, commonPolicy.BackupToolName)
- if err != nil {
- return podSpec, err
- }
- // TODO: check if pvc exists
- clusterPod, err := r.getTargetPod(reqCtx, backup, commonPolicy.Target.LabelsSelector.MatchLabels)
- if err != nil {
- return podSpec, err
- }
-
- // build pod dns string
- container := corev1.Container{}
- container.Name = backup.Name
- container.Command = backupTool.Spec.BackupCommands
- container.Image = backupTool.Spec.Image
- container.ImagePullPolicy = corev1.PullPolicy(viper.GetString(constant.KBImagePullPolicy))
- if container.Image == "" {
- // TODO(dsj): need determine container name to get, temporary use first container
- container.Image = clusterPod.Spec.Containers[0].Image
- }
- if backupTool.Spec.Resources != nil {
- container.Resources = *backupTool.Spec.Resources
- }
- container.VolumeMounts = clusterPod.Spec.Containers[0].VolumeMounts
-
- allowPrivilegeEscalation := false
- runAsUser := int64(0)
- container.SecurityContext = &corev1.SecurityContext{
- AllowPrivilegeEscalation: &allowPrivilegeEscalation,
- RunAsUser: &runAsUser}
-
- intctrlutil.InjectZeroResourcesLimitsIfEmpty(&container)
-
- envBackupName := corev1.EnvVar{
- Name: constant.DPBackupName,
- Value: backup.Name,
- }
-
- envBackupDir := corev1.EnvVar{
- Name: constant.DPBackupDIR,
- Value: backupPathBase + pathPrefix,
- }
-
- envDBHost := corev1.EnvVar{
- Name: constant.DPDBHost,
- Value: intctrlutil.BuildPodHostDNS(clusterPod),
- }
-
- envDPTargetPodName := corev1.EnvVar{
- Name: constant.DPTargetPodName,
- Value: clusterPod.Name,
- }
-
- container.Env = []corev1.EnvVar{envDPTargetPodName, envDBHost, envBackupName, envBackupDir}
- if commonPolicy.Target.Secret != nil {
- envDBUser := corev1.EnvVar{
- Name: constant.DPDBUser,
- ValueFrom: &corev1.EnvVarSource{
- SecretKeyRef: &corev1.SecretKeySelector{
- LocalObjectReference: corev1.LocalObjectReference{
- Name: commonPolicy.Target.Secret.Name,
- },
- Key: commonPolicy.Target.Secret.UsernameKey,
- },
- },
- }
-
- envDBPassword := corev1.EnvVar{
- Name: constant.DPDBPassword,
- ValueFrom: &corev1.EnvVarSource{
- SecretKeyRef: &corev1.SecretKeySelector{
- LocalObjectReference: corev1.LocalObjectReference{
- Name: commonPolicy.Target.Secret.Name,
- },
- Key: commonPolicy.Target.Secret.PasswordKey,
- },
- },
- }
-
- container.Env = append(container.Env, envDBUser, envDBPassword)
- }
-
- if backupPolicy.Spec.Retention != nil && backupPolicy.Spec.Retention.TTL != nil {
- ttl := backupPolicy.Spec.Retention.TTL
- container.Env = append(container.Env, corev1.EnvVar{
- Name: constant.DPTTL,
- Value: *ttl,
- })
- // one more day than the configured TTL for logfile backup
- logTTL := dataprotectionv1alpha1.AddTTL(ttl, 24)
- container.Env = append(container.Env, corev1.EnvVar{
- Name: constant.DPLogfileTTL,
- Value: logTTL,
- })
- container.Env = append(container.Env, corev1.EnvVar{
- Name: constant.DPLogfileTTLSecond,
- Value: strconv.FormatInt(int64(math.Floor(dataprotectionv1alpha1.ToDuration(&logTTL).Seconds())), 10),
- })
- }
-
- // merge env from backup tool.
- container.Env = append(container.Env, backupTool.Spec.Env...)
-
- podSpec.Containers = []corev1.Container{container}
- podSpec.Volumes = clusterPod.Spec.Volumes
- podSpec.RestartPolicy = corev1.RestartPolicyNever
-
- // mount the backup volume to the pod of backup tool
- pvcName := backup.Status.PersistentVolumeClaimName
- r.appendBackupVolumeMount(pvcName, &podSpec, &podSpec.Containers[0])
-
- // the pod of job needs to be scheduled on the same node as the workload pod, because it needs to share one pvc
- if clusterPod.Spec.NodeName != "" {
- podSpec.NodeSelector = map[string]string{
- hostNameLabelKey: clusterPod.Spec.NodeName,
- }
- }
- // ignore taints
- podSpec.Tolerations = []corev1.Toleration{
- {
- Operator: corev1.TolerationOpExists,
- },
- }
- return podSpec, nil
-}
-
-func (r *BackupReconciler) buildSnapshotPodSpec(
- reqCtx intctrlutil.RequestCtx,
- backup *dataprotectionv1alpha1.Backup,
- snapshotPolicy *dataprotectionv1alpha1.SnapshotPolicy,
- preCommand bool) (corev1.PodSpec, error) {
- podSpec := corev1.PodSpec{}
-
- clusterPod, err := r.getTargetPod(reqCtx, backup, snapshotPolicy.Target.LabelsSelector.MatchLabels)
- if err != nil {
- return podSpec, err
- }
-
- container := corev1.Container{}
- container.Name = backup.Name
- container.Command = []string{"kubectl", "exec", "-n", backup.Namespace,
- "-i", clusterPod.Name, "-c", snapshotPolicy.Hooks.ContainerName, "--", "sh", "-c"}
- if preCommand {
- container.Args = snapshotPolicy.Hooks.PreCommands
- } else {
- container.Args = snapshotPolicy.Hooks.PostCommands
- }
- container.Image = snapshotPolicy.Hooks.Image
- if container.Image == "" {
- container.Image = viper.GetString(constant.KBToolsImage)
- container.ImagePullPolicy = corev1.PullPolicy(viper.GetString(constant.KBImagePullPolicy))
- }
- allowPrivilegeEscalation := false
- runAsUser := int64(0)
- container.SecurityContext = &corev1.SecurityContext{
- AllowPrivilegeEscalation: &allowPrivilegeEscalation,
- RunAsUser: &runAsUser}
- intctrlutil.InjectZeroResourcesLimitsIfEmpty(&container)
- podSpec.Containers = []corev1.Container{container}
- podSpec.RestartPolicy = corev1.RestartPolicyNever
- podSpec.ServiceAccountName = viper.GetString("KUBEBLOCKS_SERVICEACCOUNT_NAME")
-
- if err = addTolerations(&podSpec); err != nil {
- return podSpec, err
- }
-
- return podSpec, nil
-}
-
-func (r *BackupReconciler) buildMetadataCollectionPodSpec(
- reqCtx intctrlutil.RequestCtx,
- backup *dataprotectionv1alpha1.Backup,
- commonPolicy *dataprotectionv1alpha1.CommonBackupPolicy,
- basePolicy *dataprotectionv1alpha1.BasePolicy,
- backupDestinationPath string,
- updateInfo dataprotectionv1alpha1.BackupStatusUpdate) (corev1.PodSpec, error) {
- podSpec := corev1.PodSpec{}
- targetPod, err := r.getTargetPod(reqCtx, backup, basePolicy.Target.LabelsSelector.MatchLabels)
- if err != nil {
- return podSpec, err
- }
-
- container := corev1.Container{}
- container.Name = backup.Name
- container.Command = []string{"sh", "-c"}
- var args string
- if strings.TrimSpace(updateInfo.Script) == "" && commonPolicy != nil {
- // if not specified script, patch backup status with the json string from ${BACKUP_DIR}/backup.info.
- args = "set -o errexit; set -o nounset;" +
- "backupInfo=$(cat ${BACKUP_INFO_FILE});echo \"backupInfo:${backupInfo}\";" +
- "eval kubectl -n %s patch backup %s --subresource=status --type=merge --patch '{\\\"status\\\":${backupInfo}}';"
- args = fmt.Sprintf(args, backup.Namespace, backup.Name)
- container.Env = []corev1.EnvVar{
- {Name: "BACKUP_INFO_FILE", Value: buildBackupInfoENV(backupDestinationPath)},
- }
- r.appendBackupVolumeMount(backup.Status.PersistentVolumeClaimName, &podSpec, &container)
- } else {
- args = "set -o errexit; set -o nounset;" +
- "OUTPUT=$(kubectl -n %s exec -it pod/%s -c %s -- %s);" +
- "kubectl -n %s patch backup %s --subresource=status --type=merge --patch \"%s\";"
- statusPath := "status." + updateInfo.Path
- if updateInfo.Path == "" {
- statusPath = "status"
- }
- patchJSON := generateJSON(statusPath, "$OUTPUT")
- args = fmt.Sprintf(args, targetPod.Namespace, targetPod.Name, updateInfo.ContainerName,
- updateInfo.Script, backup.Namespace, backup.Name, patchJSON)
- }
- if updateInfo.UseTargetPodServiceAccount {
- podSpec.ServiceAccountName = targetPod.Spec.ServiceAccountName
- } else {
- podSpec.ServiceAccountName = viper.GetString("KUBEBLOCKS_SERVICEACCOUNT_NAME")
- }
- intctrlutil.InjectZeroResourcesLimitsIfEmpty(&container)
- container.Args = []string{args}
- container.Image = viper.GetString(constant.KBToolsImage)
- container.ImagePullPolicy = corev1.PullPolicy(viper.GetString(constant.KBImagePullPolicy))
- podSpec.Containers = []corev1.Container{container}
- podSpec.RestartPolicy = corev1.RestartPolicyNever
- if err = addTolerations(&podSpec); err != nil {
- return podSpec, err
- }
- return podSpec, nil
+ return r.deleteExternalJobs(reqCtx, backup)
}
// getClusterObjectString gets the cluster object and convert it to string.
-func (r *BackupReconciler) getClusterObjectString(cluster *appsv1alpha1.Cluster) (*string, error) {
+func getClusterObjectString(cluster *appsv1alpha1.Cluster) (*string, error) {
// maintain only the cluster's spec and name/namespace.
newCluster := &appsv1alpha1.Cluster{
Spec: cluster.Spec,
@@ -2021,8 +640,8 @@ func (r *BackupReconciler) getClusterObjectString(cluster *appsv1alpha1.Cluster)
}
// setClusterSnapshotAnnotation sets the snapshot of cluster to the backup's annotations.
-func (r *BackupReconciler) setClusterSnapshotAnnotation(backup *dataprotectionv1alpha1.Backup, cluster *appsv1alpha1.Cluster) error {
- clusterString, err := r.getClusterObjectString(cluster)
+func setClusterSnapshotAnnotation(backup *dpv1alpha1.Backup, cluster *appsv1alpha1.Cluster) error {
+ clusterString, err := getClusterObjectString(cluster)
if err != nil {
return err
}
diff --git a/controllers/dataprotection/backup_controller_test.go b/controllers/dataprotection/backup_controller_test.go
index 1e069cdd350..9604e353709 100644
--- a/controllers/dataprotection/backup_controller_test.go
+++ b/controllers/dataprotection/backup_controller_test.go
@@ -20,17 +20,14 @@ along with this program. If not, see .
package dataprotection
import (
- "fmt"
"time"
. "github.com/onsi/ginkgo/v2"
. "github.com/onsi/gomega"
- "github.com/ghodss/yaml"
- snapshotv1 "github.com/kubernetes-csi/external-snapshotter/client/v6/apis/volumesnapshot/v1"
+ vsv1 "github.com/kubernetes-csi/external-snapshotter/client/v6/apis/volumesnapshot/v1"
batchv1 "k8s.io/api/batch/v1"
corev1 "k8s.io/api/core/v1"
- "k8s.io/apimachinery/pkg/api/resource"
metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
"k8s.io/apimachinery/pkg/types"
"sigs.k8s.io/controller-runtime/pkg/client"
@@ -39,25 +36,16 @@ import (
dpv1alpha1 "github.com/apecloud/kubeblocks/apis/dataprotection/v1alpha1"
storagev1alpha1 "github.com/apecloud/kubeblocks/apis/storage/v1alpha1"
"github.com/apecloud/kubeblocks/internal/constant"
- intctrlutil "github.com/apecloud/kubeblocks/internal/controllerutil"
+ dpbackup "github.com/apecloud/kubeblocks/internal/dataprotection/backup"
+ dptypes "github.com/apecloud/kubeblocks/internal/dataprotection/types"
+ dputils "github.com/apecloud/kubeblocks/internal/dataprotection/utils"
"github.com/apecloud/kubeblocks/internal/generics"
testapps "github.com/apecloud/kubeblocks/internal/testutil/apps"
+ testdp "github.com/apecloud/kubeblocks/internal/testutil/dataprotection"
viper "github.com/apecloud/kubeblocks/internal/viperx"
)
var _ = Describe("Backup Controller test", func() {
- const clusterName = "wesql-cluster"
- const componentName = "replicasets-primary"
- const containerName = "mysql"
- const backupPolicyName = "test-backup-policy"
- const backupRemotePVCName = "backup-remote-pvc"
- const defaultSchedule = "0 3 * * *"
- const defaultTTL = "7d"
- const backupName = "test-backup-job"
- const storageClassName = "test-storage-class"
-
- viper.SetDefault(constant.CfgKeyCtrlrMgrNS, testCtx.DefaultNamespace)
-
cleanEnv := func() {
// must wait till resources deleted and no longer existed before the testcases start,
// otherwise if later it needs to create some new resource objects with the same name,
@@ -68,181 +56,143 @@ var _ = Describe("Backup Controller test", func() {
// delete rest mocked objects
inNS := client.InNamespace(testCtx.DefaultNamespace)
ml := client.HasLabels{testCtx.TestObjLabelKey}
- testapps.ClearResources(&testCtx, generics.BackupToolSignature, ml)
+
// namespaced
testapps.ClearResources(&testCtx, generics.ClusterSignature, inNS, ml)
testapps.ClearResources(&testCtx, generics.PodSignature, inNS, ml)
- testapps.ClearResources(&testCtx, generics.BackupPolicySignature, inNS, ml)
testapps.ClearResourcesWithRemoveFinalizerOption(&testCtx, generics.BackupSignature, true, inNS)
+
+ // wait all backup to be deleted, otherwise the controller maybe create
+ // job to delete the backup between the ClearResources function delete
+ // the job and get the job list, resulting the ClearResources panic.
+ Eventually(testapps.List(&testCtx, generics.BackupSignature, inNS)).Should(HaveLen(0))
+
+ testapps.ClearResourcesWithRemoveFinalizerOption(&testCtx, generics.BackupPolicySignature, true, inNS)
testapps.ClearResourcesWithRemoveFinalizerOption(&testCtx, generics.JobSignature, true, inNS)
- testapps.ClearResources(&testCtx, generics.CronJobSignature, inNS, ml)
testapps.ClearResourcesWithRemoveFinalizerOption(&testCtx, generics.PersistentVolumeClaimSignature, true, inNS)
+
// non-namespaced
- testapps.ClearResources(&testCtx, generics.BackupToolSignature, ml)
+ testapps.ClearResourcesWithRemoveFinalizerOption(&testCtx, generics.ActionSetSignature, true, ml)
testapps.ClearResources(&testCtx, generics.StorageClassSignature, ml)
testapps.ClearResourcesWithRemoveFinalizerOption(&testCtx, generics.BackupRepoSignature, true, ml)
testapps.ClearResources(&testCtx, generics.StorageProviderSignature, ml)
}
- var nodeName string
- var pvcName string
- var cluster *appsv1alpha1.Cluster
+
+ var clusterInfo *testdp.BackupClusterInfo
BeforeEach(func() {
cleanEnv()
- viper.Set(constant.CfgKeyCtrlrMgrNS, testCtx.DefaultNamespace)
- By("mock a cluster")
- cluster = testapps.NewClusterFactory(testCtx.DefaultNamespace, clusterName,
- "test-cd", "test-cv").Create(&testCtx).GetObject()
- podGenerateName := clusterName + "-" + componentName
- By("By mocking a storage class")
- _ = testapps.CreateStorageClass(&testCtx, storageClassName, true)
-
- By("By mocking a pvc belonging to the pod")
- pvc := testapps.NewPersistentVolumeClaimFactory(
- testCtx.DefaultNamespace, "data-"+podGenerateName+"-0", clusterName, componentName, "data").
- SetStorage("1Gi").
- SetStorageClass(storageClassName).
- Create(&testCtx).GetObject()
- pvcName = pvc.Name
-
- By("By mocking a pvc belonging to the pod2")
- pvc2 := testapps.NewPersistentVolumeClaimFactory(
- testCtx.DefaultNamespace, "data-"+podGenerateName+"-1", clusterName, componentName, "data").
- SetStorage("1Gi").
- SetStorageClass(storageClassName).
- Create(&testCtx).GetObject()
-
- By("By mocking a pod belonging to the statefulset")
- volume := corev1.Volume{Name: pvc.Name, VolumeSource: corev1.VolumeSource{
- PersistentVolumeClaim: &corev1.PersistentVolumeClaimVolumeSource{ClaimName: pvc.Name}}}
- pod := testapps.NewPodFactory(testCtx.DefaultNamespace, podGenerateName+"-0").
- AddAppInstanceLabel(clusterName).
- AddRoleLabel("leader").
- AddAppComponentLabel(componentName).
- AddContainer(corev1.Container{Name: containerName, Image: testapps.ApeCloudMySQLImage}).
- AddVolume(volume).
- Create(&testCtx).GetObject()
- nodeName = pod.Spec.NodeName
-
- By("By mocking a pod 2 belonging to the statefulset")
- volume2 := corev1.Volume{Name: pvc2.Name, VolumeSource: corev1.VolumeSource{
- PersistentVolumeClaim: &corev1.PersistentVolumeClaimVolumeSource{ClaimName: pvc2.Name}}}
- _ = testapps.NewPodFactory(testCtx.DefaultNamespace, podGenerateName+"-1").
- AddAppInstanceLabel(clusterName).
- AddAppComponentLabel(componentName).
- AddContainer(corev1.Container{Name: containerName, Image: testapps.ApeCloudMySQLImage}).
- AddVolume(volume2).
- Create(&testCtx).GetObject()
+ clusterInfo = testdp.NewFakeCluster(&testCtx)
})
AfterEach(func() {
cleanEnv()
- viper.Set(constant.CfgKeyCtrlrMgrNS, testCtx.DefaultNamespace)
})
When("with default settings", func() {
var (
- backupTool *dpv1alpha1.BackupTool
backupPolicy *dpv1alpha1.BackupPolicy
+ repoPVCName string
+ cluster *appsv1alpha1.Cluster
+ pvcName string
+ targetPod *corev1.Pod
)
+
BeforeEach(func() {
- By("By creating a backupTool")
- backupTool = testapps.CreateCustomizedObj(&testCtx, "backup/backuptool.yaml",
- &dpv1alpha1.BackupTool{}, testapps.RandomizedObjName())
-
- By("By creating a backupPolicy from backupTool: " + backupTool.Name)
- backupPolicy = testapps.NewBackupPolicyFactory(testCtx.DefaultNamespace, backupPolicyName).
- SetTTL(defaultTTL).
- AddSnapshotPolicy().
- SetSchedule(defaultSchedule, true).
- AddMatchLabels(constant.AppInstanceLabelKey, clusterName).
- AddMatchLabels(constant.RoleLabelKey, "leader").
- SetTargetSecretName(clusterName).
- AddHookPreCommand("touch /data/mysql/.restore;sync").
- AddHookPostCommand("rm -f /data/mysql/.restore;sync").
- AddDataFilePolicy().
- SetBackupStatusUpdates([]dpv1alpha1.BackupStatusUpdate{
- {
- UpdateStage: dpv1alpha1.POST,
- },
- }).
- SetBackupToolName(backupTool.Name).
- AddMatchLabels(constant.AppInstanceLabelKey, clusterName).
- AddMatchLabels(constant.RoleLabelKey, "leader").
- SetTargetSecretName(clusterName).
- SetPVC(backupRemotePVCName).
- AddMatchLabels(constant.AppInstanceLabelKey, clusterName).
- AddLogfilePolicy().
- SetSchedule(defaultSchedule, true).
- SetPVC(backupRemotePVCName).
- SetBackupToolName(backupTool.Name).
- AddMatchLabels(constant.AppInstanceLabelKey, clusterName).
- Create(&testCtx).GetObject()
+ By("creating an actionSet")
+ actionSet := testdp.NewFakeActionSet(&testCtx)
+
+ By("creating storage provider")
+ _ = testdp.NewFakeStorageProvider(&testCtx, nil)
+
+ By("creating backup repo")
+ _, repoPVCName = testdp.NewFakeBackupRepo(&testCtx, nil)
+
+ By("creating a backupPolicy from actionSet: " + actionSet.Name)
+ backupPolicy = testdp.NewFakeBackupPolicy(&testCtx, nil)
+
+ cluster = clusterInfo.Cluster
+ pvcName = clusterInfo.TargetPVC
+ targetPod = clusterInfo.TargetPod
})
- Context("creates a datafile backup", func() {
- var backupKey types.NamespacedName
- BeforeEach(func() {
- // set datafile backup relies on logfile
- Expect(testapps.ChangeObj(&testCtx, backupTool, func(tmpObj *dpv1alpha1.BackupTool) {
- tmpObj.Spec.Physical.RelyOnLogfile = true
- })).Should(Succeed())
+ Context("creates a backup", func() {
+ var (
+ backupKey types.NamespacedName
+ backup *dpv1alpha1.Backup
+ )
+
+ getJobKey := func() client.ObjectKey {
+ return client.ObjectKey{
+ Name: dpbackup.GenerateBackupJobName(backup, dpbackup.BackupDataJobNamePrefix),
+ Namespace: backup.Namespace,
+ }
+ }
- By("By creating a backup from backupPolicy: " + backupPolicyName)
- backup := testapps.NewBackupFactory(testCtx.DefaultNamespace, backupName).
- SetBackupPolicyName(backupPolicyName).
- SetBackupType(dpv1alpha1.BackupTypeDataFile).
- Create(&testCtx).GetObject()
+ BeforeEach(func() {
+ By("creating a backup from backupPolicy " + testdp.BackupPolicyName)
+ backup = testdp.NewFakeBackup(&testCtx, nil)
backupKey = client.ObjectKeyFromObject(backup)
})
It("should succeed after job completes", func() {
By("check backup status")
Eventually(testapps.CheckObj(&testCtx, backupKey, func(g Gomega, fetched *dpv1alpha1.Backup) {
- g.Expect(fetched.Status.LogFilePersistentVolumeClaimName).Should(Equal(backupRemotePVCName))
- g.Expect(fetched.Status.Manifests.BackupTool.LogFilePath).Should(ContainSubstring(getCreatedCRNameByBackupPolicy(backupPolicy, dpv1alpha1.BackupTypeLogFile)))
+ g.Expect(fetched.Status.PersistentVolumeClaimName).Should(Equal(repoPVCName))
+ g.Expect(fetched.Status.Path).Should(Equal(dpbackup.BuildBackupPath(fetched, backupPolicy.Spec.PathPrefix)))
+ g.Expect(fetched.Status.Phase).Should(Equal(dpv1alpha1.BackupPhaseRunning))
})).Should(Succeed())
- By("Check backup job's nodeName equals pod's nodeName")
- Eventually(testapps.CheckObj(&testCtx, backupKey, func(g Gomega, fetched *batchv1.Job) {
- g.Expect(fetched.Spec.Template.Spec.NodeSelector[hostNameLabelKey]).To(Equal(nodeName))
+ By("check backup job's nodeName equals pod's nodeName")
+ Eventually(testapps.CheckObj(&testCtx, getJobKey(), func(g Gomega, fetched *batchv1.Job) {
+ g.Expect(fetched.Spec.Template.Spec.NodeSelector[corev1.LabelHostname]).To(Equal(targetPod.Spec.NodeName))
})).Should(Succeed())
- patchK8sJobStatus(backupKey, batchv1.JobComplete)
+ testdp.PatchK8sJobStatus(&testCtx, getJobKey(), batchv1.JobComplete)
+
+ By("backup job should have completed")
+ Eventually(testapps.CheckObj(&testCtx, getJobKey(), func(g Gomega, fetched *batchv1.Job) {
+ _, finishedType, _ := dputils.IsJobFinished(fetched)
+ g.Expect(finishedType).To(Equal(batchv1.JobComplete))
+ })).Should(Succeed())
- By("Check backup job completed")
+ By("backup should have completed")
Eventually(testapps.CheckObj(&testCtx, backupKey, func(g Gomega, fetched *dpv1alpha1.Backup) {
- g.Expect(fetched.Status.Phase).To(Equal(dpv1alpha1.BackupCompleted))
- g.Expect(fetched.Status.SourceCluster).Should(Equal(clusterName))
- g.Expect(fetched.Labels[constant.DataProtectionLabelClusterUIDKey]).Should(Equal(string(cluster.UID)))
- g.Expect(fetched.Labels[constant.AppInstanceLabelKey]).Should(Equal(clusterName))
- g.Expect(fetched.Labels[constant.KBAppComponentLabelKey]).Should(Equal(componentName))
+ g.Expect(fetched.Status.Phase).To(Equal(dpv1alpha1.BackupPhaseCompleted))
+ g.Expect(fetched.Labels[dptypes.DataProtectionLabelClusterUIDKey]).Should(Equal(string(cluster.UID)))
+ g.Expect(fetched.Labels[constant.AppInstanceLabelKey]).Should(Equal(testdp.ClusterName))
+ g.Expect(fetched.Labels[constant.KBAppComponentLabelKey]).Should(Equal(testdp.ComponentName))
g.Expect(fetched.Annotations[constant.ClusterSnapshotAnnotationKey]).ShouldNot(BeEmpty())
})).Should(Succeed())
- By("Check backup job is deleted after completed")
- Eventually(testapps.CheckObjExists(&testCtx, backupKey, &batchv1.Job{}, false)).Should(Succeed())
+ By("backup job should be deleted after backup completed")
+ Eventually(testapps.CheckObjExists(&testCtx, getJobKey(), &batchv1.Job{}, false)).Should(Succeed())
})
It("should fail after job fails", func() {
- patchK8sJobStatus(backupKey, batchv1.JobFailed)
+ testdp.PatchK8sJobStatus(&testCtx, getJobKey(), batchv1.JobFailed)
+
+ By("check backup job failed")
+ Eventually(testapps.CheckObj(&testCtx, getJobKey(), func(g Gomega, fetched *batchv1.Job) {
+ _, finishedType, _ := dputils.IsJobFinished(fetched)
+ g.Expect(finishedType).To(Equal(batchv1.JobFailed))
+ })).Should(Succeed())
- By("Check backup job failed")
+ By("check backup failed")
Eventually(testapps.CheckObj(&testCtx, backupKey, func(g Gomega, fetched *dpv1alpha1.Backup) {
- g.Expect(fetched.Status.Phase).To(Equal(dpv1alpha1.BackupFailed))
+ g.Expect(fetched.Status.Phase).To(Equal(dpv1alpha1.BackupPhaseFailed))
})).Should(Succeed())
})
})
- Context("deletes a datafile backup", func() {
- var backupKey types.NamespacedName
- var backup *dpv1alpha1.Backup
+ Context("deletes a backup", func() {
+ var (
+ backupKey types.NamespacedName
+ backup *dpv1alpha1.Backup
+ )
BeforeEach(func() {
- By("creating a backup from backupPolicy: " + backupPolicyName)
- backup = testapps.NewBackupFactory(testCtx.DefaultNamespace, backupName).
- SetBackupPolicyName(backupPolicyName).
- SetBackupType(dpv1alpha1.BackupTypeDataFile).
- Create(&testCtx).GetObject()
+ By("creating a backup from backupPolicy " + testdp.BackupPolicyName)
+ backup = testdp.NewFakeBackup(&testCtx, nil)
backupKey = client.ObjectKeyFromObject(backup)
By("waiting for finalizers to be added")
@@ -250,15 +200,11 @@ var _ = Describe("Backup Controller test", func() {
g.Expect(backup.GetFinalizers()).ToNot(BeEmpty())
})).Should(Succeed())
- By("setting backup file path")
+ By("setting backup status")
Eventually(testapps.ChangeObjStatus(&testCtx, backup, func() {
- if backup.Status.Manifests == nil {
- backup.Status.Manifests = &dpv1alpha1.ManifestsStatus{}
- }
- if backup.Status.Manifests.BackupTool == nil {
- backup.Status.Manifests.BackupTool = &dpv1alpha1.BackupToolManifestsStatus{}
+ if backup.Status.PersistentVolumeClaimName == "" {
+ backup.Status.PersistentVolumeClaimName = repoPVCName
}
- backup.Status.Manifests.BackupTool.FilePath = "/" + backupName
backup.Status.StartTimestamp = &metav1.Time{Time: time.Now()}
})).Should(Succeed())
})
@@ -268,51 +214,47 @@ var _ = Describe("Backup Controller test", func() {
testapps.DeleteObject(&testCtx, backupKey, &dpv1alpha1.Backup{})
By("checking new created Job")
- jobKey := buildDeleteBackupFilesJobNamespacedName(backup)
+ jobKey := dpbackup.BuildDeleteBackupFilesJobKey(backup)
job := &batchv1.Job{}
Eventually(testapps.CheckObjExists(&testCtx, jobKey,
job, true)).Should(Succeed())
- volumeName := "backup-" + backupRemotePVCName
+ volumeName := dpbackup.GenerateBackupRepoVolumeName(repoPVCName)
Eventually(testapps.CheckObj(&testCtx, jobKey, func(g Gomega, job *batchv1.Job) {
Expect(job.Spec.Template.Spec.Volumes).
Should(ContainElement(corev1.Volume{
Name: volumeName,
VolumeSource: corev1.VolumeSource{
PersistentVolumeClaim: &corev1.PersistentVolumeClaimVolumeSource{
- ClaimName: backupRemotePVCName,
+ ClaimName: repoPVCName,
},
},
}))
Expect(job.Spec.Template.Spec.Containers[0].VolumeMounts).
Should(ContainElement(corev1.VolumeMount{
Name: volumeName,
- MountPath: backupPathBase,
+ MountPath: dpbackup.RepoVolumeMountPath,
}))
})).Should(Succeed())
- By("checking Backup object, it should not be deleted")
+ By("checking backup object, it should not be deleted")
Eventually(testapps.CheckObjExists(&testCtx, backupKey,
&dpv1alpha1.Backup{}, true)).Should(Succeed())
- By("mock job for deletion to Failed, backup should not be deleted")
- Expect(testapps.ChangeObjStatus(&testCtx, job, func() {
- job.Status.Conditions = []batchv1.JobCondition{
- {
- Type: batchv1.JobFailed,
- },
- }
- })).Should(Succeed())
+ By("mock job for deletion to failed, backup should not be deleted")
+ testdp.ReplaceK8sJobStatus(&testCtx, jobKey, batchv1.JobFailed)
Eventually(testapps.CheckObjExists(&testCtx, backupKey,
&dpv1alpha1.Backup{}, true)).Should(Succeed())
By("mock job for deletion to completed, backup should be deleted")
- Expect(testapps.ChangeObjStatus(&testCtx, job, func() {
- job.Status.Conditions = []batchv1.JobCondition{
- {
- Type: batchv1.JobComplete,
- },
- }
+ testdp.ReplaceK8sJobStatus(&testCtx, jobKey, batchv1.JobComplete)
+
+ By("check deletion backup file job completed")
+ Eventually(testapps.CheckObj(&testCtx, jobKey, func(g Gomega, fetched *batchv1.Job) {
+ _, finishedType, _ := dputils.IsJobFinished(fetched)
+ g.Expect(finishedType).To(Equal(batchv1.JobComplete))
})).Should(Succeed())
+
+ By("check backup deleted")
Eventually(testapps.CheckObjExists(&testCtx, backupKey,
&dpv1alpha1.Backup{}, false)).Should(Succeed())
@@ -321,100 +263,53 @@ var _ = Describe("Backup Controller test", func() {
})
Context("creates a snapshot backup", func() {
- var backupKey types.NamespacedName
- var backup *dpv1alpha1.Backup
+ var (
+ backupKey types.NamespacedName
+ backup *dpv1alpha1.Backup
+ vsKey client.ObjectKey
+ )
BeforeEach(func() {
viper.Set("VOLUMESNAPSHOT", "true")
- viper.Set(constant.CfgKeyCtrlrMgrNS, "default")
- viper.Set(constant.CfgKeyCtrlrMgrAffinity,
- "{\"nodeAffinity\":{\"preferredDuringSchedulingIgnoredDuringExecution\":[{\"preference\":{\"matchExpressions\":[{\"key\":\"kb-controller\",\"operator\":\"In\",\"values\":[\"true\"]}]},\"weight\":100}]}}")
- viper.Set(constant.CfgKeyCtrlrMgrTolerations,
- "[{\"key\":\"key1\", \"operator\": \"Exists\", \"effect\": \"NoSchedule\"}]")
- viper.Set(constant.CfgKeyCtrlrMgrNodeSelector, "{\"beta.kubernetes.io/arch\":\"amd64\"}")
- snapshotBackupName := "backup-default-postgres-cluster-20230628104804"
- By("By creating a backup from backupPolicy: " + backupPolicyName)
- backup = testapps.NewBackupFactory(testCtx.DefaultNamespace, snapshotBackupName).
- SetBackupPolicyName(backupPolicyName).
- SetBackupType(dpv1alpha1.BackupTypeSnapshot).
- Create(&testCtx).GetObject()
+ By("create a backup from backupPolicy " + testdp.BackupPolicyName)
+ backup = testdp.NewFakeBackup(&testCtx, func(backup *dpv1alpha1.Backup) {
+ backup.Spec.BackupMethod = testdp.VSBackupMethodName
+ })
backupKey = client.ObjectKeyFromObject(backup)
+ vsKey = client.ObjectKey{
+ Name: dputils.GetBackupVolumeSnapshotName(backup.Name, "data"),
+ Namespace: backup.Namespace,
+ }
})
AfterEach(func() {
viper.Set("VOLUMESNAPSHOT", "false")
- viper.Set(constant.CfgKeyCtrlrMgrAffinity, "")
- viper.Set(constant.CfgKeyCtrlrMgrTolerations, "")
- viper.Set(constant.CfgKeyCtrlrMgrNodeSelector, "")
})
- It("should success after all jobs complete", func() {
- backupPolicyKey := types.NamespacedName{Name: backupPolicyName, Namespace: backupKey.Namespace}
- patchBackupPolicySpecBackupStatusUpdates(backupPolicyKey)
-
- preJobKey := types.NamespacedName{Name: generateUniqueJobName(backup, "hook-pre"), Namespace: backupKey.Namespace}
- postJobKey := types.NamespacedName{Name: generateUniqueJobName(backup, "hook-post"), Namespace: backupKey.Namespace}
- patchK8sJobStatus(preJobKey, batchv1.JobComplete)
- By("Check job tolerations")
- Eventually(testapps.CheckObj(&testCtx, preJobKey, func(g Gomega, fetched *batchv1.Job) {
- g.Expect(fetched.Spec.Template.Spec.Tolerations).ShouldNot(BeEmpty())
- g.Expect(fetched.Spec.Template.Spec.NodeSelector).ShouldNot(BeEmpty())
- g.Expect(fetched.Spec.Template.Spec.Affinity).ShouldNot(BeNil())
- g.Expect(fetched.Spec.Template.Spec.Affinity.NodeAffinity).ShouldNot(BeNil())
- })).Should(Succeed())
-
- patchVolumeSnapshotStatus(backupKey, true)
- patchK8sJobStatus(postJobKey, batchv1.JobComplete)
-
- logJobKey := types.NamespacedName{Name: generateUniqueJobName(backup, "status-0-pre"), Namespace: backupKey.Namespace}
- patchK8sJobStatus(logJobKey, batchv1.JobComplete)
+ It("should success after all volume snapshot ready", func() {
+ By("patching volumesnapshot status to ready")
+ testdp.PatchVolumeSnapshotStatus(&testCtx, vsKey, true)
- By("Check backup job completed")
- Eventually(testapps.CheckObj(&testCtx, backupKey, func(g Gomega, fetched *dpv1alpha1.Backup) {
- g.Expect(fetched.Status.Phase).To(Equal(dpv1alpha1.BackupCompleted))
- })).Should(Succeed())
-
- sizeJobKey := types.NamespacedName{Name: generateUniqueJobName(backup, "status-1-post"), Namespace: backupKey.Namespace}
- patchK8sJobStatus(sizeJobKey, batchv1.JobComplete)
-
- By("Check pre job cleaned")
- Eventually(testapps.CheckObjExists(&testCtx, preJobKey, &batchv1.Job{}, false)).Should(Succeed())
- By("Check post job cleaned")
- Eventually(testapps.CheckObjExists(&testCtx, postJobKey, &batchv1.Job{}, false)).Should(Succeed())
- By("Check if the target pod name is correct")
- Eventually(testapps.CheckObj(&testCtx, backupKey, func(g Gomega, fetched *snapshotv1.VolumeSnapshot) {
+ By("checking volume snapshot source is equal to pvc")
+ Eventually(testapps.CheckObj(&testCtx, vsKey, func(g Gomega, fetched *vsv1.VolumeSnapshot) {
g.Expect(*fetched.Spec.Source.PersistentVolumeClaimName).To(Equal(pvcName))
})).Should(Succeed())
})
- It("should fail after pre-job fails", func() {
- patchK8sJobStatus(types.NamespacedName{Name: generateUniqueJobName(backup, "hook-pre"), Namespace: backupKey.Namespace}, batchv1.JobFailed)
-
- By("Check backup job failed")
- Eventually(testapps.CheckObj(&testCtx, backupKey, func(g Gomega, fetched *dpv1alpha1.Backup) {
- g.Expect(fetched.Status.Phase).To(Equal(dpv1alpha1.BackupFailed))
- })).Should(Succeed())
- })
-
It("should fail if volumesnapshot reports error", func() {
-
- By("patching job status to pass check")
- preJobKey := types.NamespacedName{Name: generateUniqueJobName(backup, "hook-pre"), Namespace: backupKey.Namespace}
- patchK8sJobStatus(preJobKey, batchv1.JobComplete)
-
By("patching volumesnapshot status with error")
- Eventually(testapps.GetAndChangeObjStatus(&testCtx, backupKey, func(tmpVS *snapshotv1.VolumeSnapshot) {
+ Eventually(testapps.GetAndChangeObjStatus(&testCtx, vsKey, func(tmpVS *vsv1.VolumeSnapshot) {
msg := "Failed to set default snapshot class with error: some error"
- vsError := snapshotv1.VolumeSnapshotError{
+ vsError := vsv1.VolumeSnapshotError{
Message: &msg,
}
- snapStatus := snapshotv1.VolumeSnapshotStatus{Error: &vsError}
+ snapStatus := vsv1.VolumeSnapshotStatus{Error: &vsError}
tmpVS.Status = &snapStatus
})).Should(Succeed())
By("checking backup failed")
Eventually(testapps.CheckObj(&testCtx, backupKey, func(g Gomega, fetched *dpv1alpha1.Backup) {
- g.Expect(fetched.Status.Phase).To(Equal(dpv1alpha1.BackupFailed))
+ g.Expect(fetched.Status.Phase).To(Equal(dpv1alpha1.BackupPhaseFailed))
})).Should(Succeed())
})
})
@@ -428,374 +323,77 @@ var _ = Describe("Backup Controller test", func() {
// delete rest mocked objects
inNS := client.InNamespace(testCtx.DefaultNamespace)
ml := client.HasLabels{testCtx.TestObjLabelKey}
- testapps.ClearResourcesWithRemoveFinalizerOption(&testCtx, generics.PersistentVolumeClaimSignature, true, inNS, ml)
+ testapps.ClearResourcesWithRemoveFinalizerOption(&testCtx,
+ generics.PersistentVolumeClaimSignature, true, inNS, ml)
})
It("should fail when disable volumesnapshot", func() {
viper.Set("VOLUMESNAPSHOT", "false")
-
- By("By creating a backup from backupPolicy: " + backupPolicyName)
- backup := testapps.NewBackupFactory(testCtx.DefaultNamespace, backupName).
- SetBackupPolicyName(backupPolicyName).
- SetBackupType(dpv1alpha1.BackupTypeSnapshot).
- Create(&testCtx).GetObject()
+ By("creating a backup from backupPolicy " + testdp.BackupPolicyName)
+ backup := testdp.NewFakeBackup(&testCtx, func(backup *dpv1alpha1.Backup) {
+ backup.Spec.BackupMethod = testdp.VSBackupMethodName
+ })
backupKey = client.ObjectKeyFromObject(backup)
- By("Check backup job failed")
+ By("check backup failed")
Eventually(testapps.CheckObj(&testCtx, backupKey, func(g Gomega, fetched *dpv1alpha1.Backup) {
- g.Expect(fetched.Status.Phase).To(Equal(dpv1alpha1.BackupFailed))
+ g.Expect(fetched.Status.Phase).To(Equal(dpv1alpha1.BackupPhaseFailed))
})).Should(Succeed())
})
It("should fail without pvc", func() {
- By("By creating a backup from backupPolicy: " + backupPolicyName)
- backup := testapps.NewBackupFactory(testCtx.DefaultNamespace, backupName).
- SetBackupPolicyName(backupPolicyName).
- SetBackupType(dpv1alpha1.BackupTypeSnapshot).
- Create(&testCtx).GetObject()
- backupKey = client.ObjectKeyFromObject(backup)
-
- patchK8sJobStatus(types.NamespacedName{Name: generateUniqueJobName(backup, "hook-pre"), Namespace: backupKey.Namespace}, batchv1.JobComplete)
-
- By("Check backup job failed")
- Eventually(testapps.CheckObj(&testCtx, backupKey, func(g Gomega, fetched *dpv1alpha1.Backup) {
- g.Expect(fetched.Status.Phase).To(Equal(dpv1alpha1.BackupFailed))
- })).Should(Succeed())
- })
-
- })
-
- Context("creates a logfile backup", func() {
- var backupKey types.NamespacedName
-
- BeforeEach(func() {
- backupTool := testapps.CreateCustomizedObj(&testCtx, "backup/backuptool.yaml",
- &dpv1alpha1.BackupTool{}, testapps.RandomizedObjName())
- backupPolicy := testapps.NewBackupPolicyFactory(testCtx.DefaultNamespace, backupPolicyName).
- WithRandomName().
- AddLogfilePolicy().
- SetTTL("7d").
- SetSchedule("*/1 * * * *", true).
- SetBackupToolName(backupTool.Name).
- SetPVC(backupRemotePVCName).
- AddMatchLabels(constant.AppInstanceLabelKey, clusterName).
- Create(&testCtx).GetObject()
- By("By creating a backup from backupPolicy: " + backupPolicy.Name)
- logFileBackupName := getCreatedCRNameByBackupPolicy(backupPolicy, dpv1alpha1.BackupTypeLogFile)
- backup := testapps.NewBackupFactory(testCtx.DefaultNamespace, logFileBackupName).
- SetBackupPolicyName(backupPolicy.Name).
- SetBackupType(dpv1alpha1.BackupTypeLogFile).
- Create(&testCtx).GetObject()
+ By("creating a backup from backupPolicy " + testdp.BackupPolicyName)
+ backup := testdp.NewFakeBackup(&testCtx, func(backup *dpv1alpha1.Backup) {
+ backup.Spec.BackupMethod = testdp.VSBackupMethodName
+ })
backupKey = client.ObjectKeyFromObject(backup)
- })
-
- It("should succeed", func() {
- By("Check backup job's nodeName equals pod's nodeName")
- Eventually(testapps.CheckObj(&testCtx, backupKey, func(g Gomega, fetched *batchv1.Job) {
- g.Expect(fetched.Spec.Template.Spec.NodeSelector[hostNameLabelKey]).To(Equal(nodeName))
- })).Should(Succeed())
-
- patchK8sJobStatus(backupKey, batchv1.JobComplete)
-
- By("Check backup job completed")
- Eventually(testapps.CheckObj(&testCtx, backupKey, func(g Gomega, fetched *dpv1alpha1.Backup) {
- g.Expect(fetched.Status.Phase).To(Equal(dpv1alpha1.BackupCompleted))
- g.Expect(fetched.Status.SourceCluster).Should(Equal(clusterName))
- g.Expect(fetched.Labels[constant.DataProtectionLabelClusterUIDKey]).Should(Equal(string(cluster.UID)))
- g.Expect(fetched.Labels[constant.AppInstanceLabelKey]).Should(Equal(clusterName))
- g.Expect(fetched.Labels[constant.KBAppComponentLabelKey]).Should(Equal(componentName))
- g.Expect(fetched.Annotations[constant.ClusterSnapshotAnnotationKey]).ShouldNot(BeEmpty())
- })).Should(Succeed())
-
- By("Check backup job is deleted after completed")
- Eventually(testapps.CheckObjExists(&testCtx, backupKey, &batchv1.Job{}, false)).Should(Succeed())
- })
-
- It("should succeed if the previous job failed and the current job succeeded", func() {
- patchK8sJobStatus(backupKey, batchv1.JobFailed)
-
- By("Check backup job failed")
- Eventually(testapps.CheckObj(&testCtx, backupKey, func(g Gomega, fetched *dpv1alpha1.Backup) {
- g.Expect(fetched.Status.Phase).To(Equal(dpv1alpha1.BackupFailed))
- })).Should(Succeed())
- By("Patch backup Phase to New")
- Eventually(testapps.GetAndChangeObjStatus(&testCtx, backupKey, func(fetched *dpv1alpha1.Backup) {
- fetched.Status.Phase = dpv1alpha1.BackupNew
- })).Should(Succeed())
-
- By("Check backup job completed")
- Eventually(testapps.CheckObj(&testCtx, backupKey, func(g Gomega, fetched *dpv1alpha1.Backup) {
- g.Expect(fetched.Status.Phase).To(Equal(dpv1alpha1.BackupInProgress))
- })).Should(Succeed())
- patchK8sJobStatus(backupKey, batchv1.JobComplete)
+ By("check backup failed")
Eventually(testapps.CheckObj(&testCtx, backupKey, func(g Gomega, fetched *dpv1alpha1.Backup) {
- g.Expect(fetched.Status.Phase).To(Equal(dpv1alpha1.BackupCompleted))
+ g.Expect(fetched.Status.Phase).To(Equal(dpv1alpha1.BackupPhaseFailed))
})).Should(Succeed())
})
})
})
- When("with backupTool resources", func() {
- Context("creates a datafile backup", func() {
- var backupKey types.NamespacedName
- var backupPolicy *dpv1alpha1.BackupPolicy
- var pathPrefix = "/mysql/backup"
- createBackup := func(backupName string) {
- By("By creating a backup from backupPolicy: " + backupPolicyName)
- backup := testapps.NewBackupFactory(testCtx.DefaultNamespace, backupName).
- SetBackupPolicyName(backupPolicyName).
- SetBackupType(dpv1alpha1.BackupTypeDataFile).
- Create(&testCtx).GetObject()
- backupKey = client.ObjectKeyFromObject(backup)
- }
-
- BeforeEach(func() {
- viper.SetDefault(constant.CfgKeyBackupPVCStorageClass, "")
- By("By creating a backupTool")
- backupTool := testapps.CreateCustomizedObj(&testCtx, "backup/backuptool.yaml",
- &dpv1alpha1.BackupTool{}, testapps.RandomizedObjName(),
- func(backupTool *dpv1alpha1.BackupTool) {
- backupTool.Spec.Resources = nil
- })
-
- By("By creating a backupPolicy from backupTool: " + backupTool.Name)
- backupPolicy = testapps.NewBackupPolicyFactory(testCtx.DefaultNamespace, backupPolicyName).
- AddAnnotations(constant.BackupDataPathPrefixAnnotationKey, pathPrefix).
- AddDataFilePolicy().
- SetBackupToolName(backupTool.Name).
- SetSchedule(defaultSchedule, true).
- SetTTL(defaultTTL).
- AddMatchLabels(constant.AppInstanceLabelKey, clusterName).
- SetTargetSecretName(clusterName).
- SetPVC(backupRemotePVCName).
- Create(&testCtx).GetObject()
-
- })
-
- It("should succeed after job completes", func() {
- createBackup(backupName)
- patchK8sJobStatus(backupKey, batchv1.JobComplete)
- By("Check backup job completed")
- Eventually(testapps.CheckObj(&testCtx, backupKey, func(g Gomega, fetched *dpv1alpha1.Backup) {
- g.Expect(fetched.Status.Phase).To(Equal(dpv1alpha1.BackupCompleted))
- g.Expect(fetched.Status.Manifests.BackupTool.FilePath).To(Equal(fmt.Sprintf("/%s%s/%s", backupKey.Namespace, pathPrefix, backupKey.Name)))
- })).Should(Succeed())
- })
-
- It("creates pvc if the specified pvc not exists", func() {
- createBackup(backupName)
- By("Check pvc created by backup controller")
- Eventually(testapps.CheckObjExists(&testCtx, types.NamespacedName{
- Name: backupRemotePVCName,
- Namespace: testCtx.DefaultNamespace,
- }, &corev1.PersistentVolumeClaim{}, true)).Should(Succeed())
- })
-
- It("creates pvc if the specified pvc not exists", func() {
- By("set persistentVolumeConfigmap")
- configMapName := "pv-template-configmap"
- Expect(testapps.ChangeObj(&testCtx, backupPolicy, func(tmpObj *dpv1alpha1.BackupPolicy) {
- tmpObj.Spec.Datafile.PersistentVolumeClaim.PersistentVolumeConfigMap = &dpv1alpha1.PersistentVolumeConfigMap{
- Name: configMapName,
- Namespace: testCtx.DefaultNamespace,
- }
- })).Should(Succeed())
-
- By("create backup with non existent configmap of pv template")
- createBackup(backupName)
- Eventually(testapps.CheckObj(&testCtx, backupKey, func(g Gomega, fetched *dpv1alpha1.Backup) {
- g.Expect(fetched.Status.Phase).To(Equal(dpv1alpha1.BackupFailed))
- g.Expect(fetched.Status.FailureReason).To(ContainSubstring(fmt.Sprintf(`ConfigMap "%s" not found`, configMapName)))
- })).Should(Succeed())
- configMap := &corev1.ConfigMap{
- ObjectMeta: metav1.ObjectMeta{
- Name: configMapName,
- Namespace: testCtx.DefaultNamespace,
- },
- Data: map[string]string{},
- }
- Expect(testCtx.CreateObj(ctx, configMap)).Should(Succeed())
-
- By("create backup with the configmap not contains the key 'persistentVolume'")
- createBackup(backupName + "1")
- Eventually(testapps.CheckObj(&testCtx, backupKey, func(g Gomega, fetched *dpv1alpha1.Backup) {
- g.Expect(fetched.Status.Phase).To(Equal(dpv1alpha1.BackupFailed))
- g.Expect(fetched.Status.FailureReason).To(ContainSubstring("the persistentVolume template is empty in the configMap"))
- })).Should(Succeed())
-
- By("create backup with the configmap contains the key 'persistentVolume'")
- Expect(testapps.ChangeObj(&testCtx, configMap, func(tmpObj *corev1.ConfigMap) {
- pv := corev1.PersistentVolume{
- Spec: corev1.PersistentVolumeSpec{
- AccessModes: []corev1.PersistentVolumeAccessMode{
- corev1.ReadWriteMany,
- },
- Capacity: corev1.ResourceList{
- corev1.ResourceStorage: resource.MustParse("1Gi"),
- },
- PersistentVolumeReclaimPolicy: corev1.PersistentVolumeReclaimRetain,
- PersistentVolumeSource: corev1.PersistentVolumeSource{
- CSI: &corev1.CSIPersistentVolumeSource{
- Driver: "kubeblocks.com",
- FSType: "ext4",
- VolumeHandle: pvcName,
- },
- },
- },
- }
- pvString, _ := yaml.Marshal(pv)
- tmpObj.Data = map[string]string{
- "persistentVolume": string(pvString),
- }
- })).Should(Succeed())
- createBackup(backupName + "2")
- Eventually(testapps.CheckObj(&testCtx, backupKey, func(g Gomega, fetched *dpv1alpha1.Backup) {
- g.Expect(fetched.Status.Phase).To(Equal(dpv1alpha1.BackupInProgress))
- })).Should(Succeed())
-
- By("check pvc and pv created by backup controller")
- Eventually(testapps.CheckObjExists(&testCtx, types.NamespacedName{
- Name: backupRemotePVCName,
- Namespace: testCtx.DefaultNamespace,
- }, &corev1.PersistentVolumeClaim{}, true)).Should(Succeed())
- Eventually(testapps.CheckObjExists(&testCtx, types.NamespacedName{
- Name: backupRemotePVCName + "-" + testCtx.DefaultNamespace,
- Namespace: testCtx.DefaultNamespace,
- }, &corev1.PersistentVolume{}, true)).Should(Succeed())
-
- })
- })
- })
When("with exceptional settings", func() {
- Context("creates a backup with non existent backup policy", func() {
+ Context("creates a backup with non-existent backup policy", func() {
var backupKey types.NamespacedName
BeforeEach(func() {
- By("By creating a backup from backupPolicy: " + backupPolicyName)
- backup := testapps.NewBackupFactory(testCtx.DefaultNamespace, backupName).
- SetBackupPolicyName(backupPolicyName).
- SetBackupType(dpv1alpha1.BackupTypeDataFile).
- Create(&testCtx).GetObject()
+ By("creating a backup from backupPolicy " + testdp.BackupPolicyName)
+ backup := testdp.NewFakeBackup(&testCtx, nil)
backupKey = client.ObjectKeyFromObject(backup)
})
- It("Should fail", func() {
- By("Check backup status failed")
+ It("should fail", func() {
+ By("check backup status failed")
Eventually(testapps.CheckObj(&testCtx, backupKey, func(g Gomega, fetched *dpv1alpha1.Backup) {
- g.Expect(fetched.Status.Phase).To(Equal(dpv1alpha1.BackupFailed))
+ g.Expect(fetched.Status.Phase).To(Equal(dpv1alpha1.BackupPhaseFailed))
})).Should(Succeed())
})
})
})
- When("with logfile backup", func() {
- Context("test logfile backup", func() {
- It("testing the legality of logfile backup ", func() {
- By("init test resources")
- // mock a backupTool
- backupTool := createStatefulKindBackupTool()
- backupPolicy := testapps.NewBackupPolicyFactory(testCtx.DefaultNamespace, backupPolicyName).
- AddLogfilePolicy().
- SetTTL("7d").
- SetSchedule("*/1 * * * *", false).
- SetBackupToolName(backupTool.Name).
- SetPVC(backupRemotePVCName).
- AddMatchLabels(constant.AppInstanceLabelKey, clusterName).
- Create(&testCtx).GetObject()
- By("create logfile backup with a invalid name, expect error")
- backup := testapps.NewBackupFactory(testCtx.DefaultNamespace, "test-logfile").
- SetBackupPolicyName(backupPolicyName).
- SetBackupType(dpv1alpha1.BackupTypeLogFile).
- Create(&testCtx).GetObject()
- Eventually(testapps.CheckObj(&testCtx, client.ObjectKeyFromObject(backup), func(g Gomega, backup *dpv1alpha1.Backup) {
- g.Expect(backup.Status.Phase).Should(Equal(dpv1alpha1.BackupFailed))
- expectErr := intctrlutil.NewInvalidLogfileBackupName(backupPolicyName)
- g.Expect(backup.Status.FailureReason).Should(Equal(expectErr.Error()))
- })).Should(Succeed())
- By("update logfile backup with valid name, but the schedule is disabled, expect error")
- backup = testapps.NewBackupFactory(testCtx.DefaultNamespace, getCreatedCRNameByBackupPolicy(backupPolicy, dpv1alpha1.BackupTypeLogFile)).
- SetBackupPolicyName(backupPolicyName).
- SetBackupType(dpv1alpha1.BackupTypeLogFile).
- Create(&testCtx).GetObject()
- Eventually(testapps.CheckObj(&testCtx, client.ObjectKeyFromObject(backup), func(g Gomega, backup *dpv1alpha1.Backup) {
- g.Expect(backup.Status.Phase).Should(Equal(dpv1alpha1.BackupFailed))
- expectErr := intctrlutil.NewBackupScheduleDisabled(string(dpv1alpha1.BackupTypeLogFile), backupPolicyName)
- g.Expect(backup.Status.FailureReason).Should(Equal(expectErr.Error()))
- })).Should(Succeed())
- })
- })
- })
When("with backup repo", func() {
- var sp *storagev1alpha1.StorageProvider
- var repo *dpv1alpha1.BackupRepo
- var repoPVCName string
- var backupTool *dpv1alpha1.BackupTool
-
- createBackupPolicy := func(pvcName string, repoName string) *dpv1alpha1.BackupPolicy {
- builder := testapps.NewBackupPolicyFactory(testCtx.DefaultNamespace, backupPolicyName).
- AddDataFilePolicy().
- SetPVC(pvcName).
- SetBackupRepo(repoName).
- SetBackupToolName(backupTool.Name).
- AddMatchLabels(constant.AppInstanceLabelKey, clusterName)
- return builder.Create(&testCtx).GetObject()
- }
-
- createBackup := func(policy *dpv1alpha1.BackupPolicy, change func(*dpv1alpha1.Backup)) *dpv1alpha1.Backup {
- if change == nil {
- change = func(*dpv1alpha1.Backup) {} // set nop
- }
- backup := testapps.NewBackupFactory(testCtx.DefaultNamespace, backupName).
- SetBackupPolicyName(backupPolicyName).
- SetBackupType(dpv1alpha1.BackupTypeDataFile).
- Apply(change).
- Create(&testCtx).GetObject()
- return backup
- }
-
- createStorageProvider := func() *storagev1alpha1.StorageProvider {
- sp := testapps.CreateCustomizedObj(&testCtx, "backup/storageprovider.yaml",
- &storagev1alpha1.StorageProvider{})
- // the storage provider controller is not running, so set the status manually
- Expect(testapps.ChangeObjStatus(&testCtx, sp, func() {
- sp.Status.Phase = storagev1alpha1.StorageProviderReady
- })).Should(Succeed())
- return sp
- }
-
- createRepo := func(change func(repo *dpv1alpha1.BackupRepo)) (*dpv1alpha1.BackupRepo, string) {
- repo := testapps.CreateCustomizedObj(&testCtx, "backup/backuprepo.yaml",
- &dpv1alpha1.BackupRepo{}, func(obj *dpv1alpha1.BackupRepo) {
- obj.Spec.StorageProviderRef = sp.Name
- if change != nil {
- change(obj)
- }
- })
- var repoPVCName string
- Eventually(testapps.CheckObj(&testCtx, client.ObjectKeyFromObject(repo), func(g Gomega, repo *dpv1alpha1.BackupRepo) {
- g.Expect(repo.Status.Phase).Should(BeEquivalentTo(dpv1alpha1.BackupRepoReady))
- g.Expect(repo.Status.BackupPVCName).ShouldNot(BeEmpty())
- repoPVCName = repo.Status.BackupPVCName
- })).Should(Succeed())
- return repo, repoPVCName
- }
+ var (
+ repoPVCName string
+ sp *storagev1alpha1.StorageProvider
+ repo *dpv1alpha1.BackupRepo
+ )
BeforeEach(func() {
By("creating backup repo")
- sp = createStorageProvider()
- repo, repoPVCName = createRepo(nil)
-
- By("creating backup tool")
- backupTool = testapps.CreateCustomizedObj(&testCtx, "backup/backuptool.yaml",
- &dpv1alpha1.BackupTool{}, testapps.RandomizedObjName())
+ sp = testdp.NewFakeStorageProvider(&testCtx, nil)
+ repo, repoPVCName = testdp.NewFakeBackupRepo(&testCtx, nil)
- viper.SetDefault(constant.CfgKeyBackupPVCName, "")
+ By("creating actionSet")
+ _ = testdp.NewFakeActionSet(&testCtx)
})
Context("explicitly specify backup repo", func() {
It("should use the backup repo specified in the policy", func() {
By("creating backup policy and backup")
- policy := createBackupPolicy("", repo.Name)
- backup := createBackup(policy, nil)
+ _ = testdp.NewFakeBackupPolicy(&testCtx, nil)
+ backup := testdp.NewFakeBackup(&testCtx, nil)
By("checking backup, it should use the PVC from the backup repo")
Eventually(testapps.CheckObj(&testCtx, client.ObjectKeyFromObject(backup), func(g Gomega, backup *dpv1alpha1.Backup) {
g.Expect(backup.Status.PersistentVolumeClaimName).Should(BeEquivalentTo(repoPVCName))
@@ -804,10 +402,14 @@ var _ = Describe("Backup Controller test", func() {
It("should use the backup repo specified in the backup object", func() {
By("creating a second backup repo")
- repo2, repoPVCName2 := createRepo(nil)
+ repo2, repoPVCName2 := testdp.NewFakeBackupRepo(&testCtx, func(repo *dpv1alpha1.BackupRepo) {
+ repo.Name += "2"
+ })
By("creating backup policy and backup")
- policy := createBackupPolicy("", repo.Name)
- backup := createBackup(policy, func(backup *dpv1alpha1.Backup) {
+ _ = testdp.NewFakeBackupPolicy(&testCtx, func(backupPolicy *dpv1alpha1.BackupPolicy) {
+ backupPolicy.Spec.BackupRepoName = &repo.Name
+ })
+ backup := testdp.NewFakeBackup(&testCtx, func(backup *dpv1alpha1.Backup) {
if backup.Labels == nil {
backup.Labels = map[string]string{}
}
@@ -823,8 +425,10 @@ var _ = Describe("Backup Controller test", func() {
Context("default backup repo", func() {
It("should use the default backup repo if it's not specified", func() {
By("creating backup policy and backup")
- policy := createBackupPolicy("", "")
- backup := createBackup(policy, nil)
+ _ = testdp.NewFakeBackupPolicy(&testCtx, func(backupPolicy *dpv1alpha1.BackupPolicy) {
+ backupPolicy.Spec.BackupRepoName = nil
+ })
+ backup := testdp.NewFakeBackup(&testCtx, nil)
By("checking backup, it should use the PVC from the backup repo")
Eventually(testapps.CheckObj(&testCtx, client.ObjectKeyFromObject(backup), func(g Gomega, backup *dpv1alpha1.Backup) {
g.Expect(backup.Status.PersistentVolumeClaimName).Should(BeEquivalentTo(repoPVCName))
@@ -833,15 +437,17 @@ var _ = Describe("Backup Controller test", func() {
It("should associate the default backup repo with the backup object", func() {
By("creating backup policy and backup")
- policy := createBackupPolicy("", "")
- backup := createBackup(policy, nil)
+ _ = testdp.NewFakeBackupPolicy(&testCtx, func(backupPolicy *dpv1alpha1.BackupPolicy) {
+ backupPolicy.Spec.BackupRepoName = nil
+ })
+ backup := testdp.NewFakeBackup(&testCtx, nil)
By("checking backup labels")
Eventually(testapps.CheckObj(&testCtx, client.ObjectKeyFromObject(backup), func(g Gomega, backup *dpv1alpha1.Backup) {
g.Expect(backup.Labels[dataProtectionBackupRepoKey]).Should(BeEquivalentTo(repo.Name))
})).Should(Succeed())
By("creating backup2")
- backup2 := createBackup(policy, func(backup *dpv1alpha1.Backup) {
+ backup2 := testdp.NewFakeBackup(&testCtx, func(backup *dpv1alpha1.Backup) {
backup.Name += "2"
})
By("checking backup2 labels")
@@ -853,23 +459,28 @@ var _ = Describe("Backup Controller test", func() {
Context("multiple default backup repos", func() {
var repoPVCName2 string
- var policy *dpv1alpha1.BackupPolicy
BeforeEach(func() {
By("creating a second backup repo")
- sp2 := createStorageProvider()
- _, repoPVCName2 = createRepo(func(repo *dpv1alpha1.BackupRepo) {
+ sp2 := testdp.NewFakeStorageProvider(&testCtx, func(sp *storagev1alpha1.StorageProvider) {
+ sp.Name += "2"
+ })
+ _, repoPVCName2 = testdp.NewFakeBackupRepo(&testCtx, func(repo *dpv1alpha1.BackupRepo) {
+ repo.Name += "2"
repo.Spec.StorageProviderRef = sp2.Name
})
By("creating backup policy")
- policy = createBackupPolicy("", "")
+ _ = testdp.NewFakeBackupPolicy(&testCtx, func(backupPolicy *dpv1alpha1.BackupPolicy) {
+ // set backupRepoName in backupPolicy to nil to make it use the default backup repo
+ backupPolicy.Spec.BackupRepoName = nil
+ })
})
It("should fail if there are multiple default backup repos", func() {
By("creating backup")
- backup := createBackup(policy, nil)
+ backup := testdp.NewFakeBackup(&testCtx, nil)
By("checking backup, it should fail because there are multiple default backup repos")
Eventually(testapps.CheckObj(&testCtx, client.ObjectKeyFromObject(backup), func(g Gomega, backup *dpv1alpha1.Backup) {
- g.Expect(backup.Status.Phase).Should(BeEquivalentTo(dpv1alpha1.BackupFailed))
+ g.Expect(backup.Status.Phase).Should(BeEquivalentTo(dpv1alpha1.BackupPhaseFailed))
g.Expect(backup.Status.FailureReason).Should(ContainSubstring("multiple default BackupRepo found"))
})).Should(Succeed())
})
@@ -885,7 +496,7 @@ var _ = Describe("Backup Controller test", func() {
g.Expect(repo.Status.Phase).Should(BeEquivalentTo(dpv1alpha1.BackupRepoFailed))
})).Should(Succeed())
By("creating backup")
- backup := createBackup(policy, func(backup *dpv1alpha1.Backup) {
+ backup := testdp.NewFakeBackup(&testCtx, func(backup *dpv1alpha1.Backup) {
backup.Name = "second-backup"
})
By("checking backup, it should use the PVC from repo2")
@@ -897,53 +508,22 @@ var _ = Describe("Backup Controller test", func() {
})
Context("no backup repo available", func() {
- It("should fallback to the legacy PVC settings", func() {
+ It("should throw error", func() {
By("making the backup repo as non-default")
Eventually(testapps.GetAndChangeObj(&testCtx, client.ObjectKeyFromObject(repo), func(repo *dpv1alpha1.BackupRepo) {
- delete(repo.Annotations, constant.DefaultBackupRepoAnnotationKey)
+ delete(repo.Annotations, dptypes.DefaultBackupRepoAnnotationKey)
})).Should(Succeed())
By("creating backup")
- policy := createBackupPolicy("", "")
- backup := createBackup(policy, nil)
- By("checking backup, it should fail because neither the backup repo nor the legacy PVC are available")
+ _ = testdp.NewFakeBackupPolicy(&testCtx, func(backupPolicy *dpv1alpha1.BackupPolicy) {
+ backupPolicy.Spec.BackupRepoName = nil
+ })
+ backup := testdp.NewFakeBackup(&testCtx, nil)
+ By("checking backup, it should fail because the backup repo are not available")
Eventually(testapps.CheckObj(&testCtx, client.ObjectKeyFromObject(backup), func(g Gomega, backup *dpv1alpha1.Backup) {
- g.Expect(backup.Status.Phase).Should(BeEquivalentTo(dpv1alpha1.BackupFailed))
- g.Expect(backup.Status.FailureReason).Should(ContainSubstring("the persistentVolumeClaim name of spec.datafile is empty"))
+ g.Expect(backup.Status.Phase).Should(BeEquivalentTo(dpv1alpha1.BackupPhaseFailed))
+ g.Expect(backup.Status.FailureReason).Should(ContainSubstring("no default BackupRepo found"))
})).Should(Succeed())
})
})
})
})
-
-func patchK8sJobStatus(key types.NamespacedName, jobStatus batchv1.JobConditionType) {
- Eventually(testapps.GetAndChangeObjStatus(&testCtx, key, func(fetched *batchv1.Job) {
- jobCondition := batchv1.JobCondition{Type: jobStatus}
- fetched.Status.Conditions = append(fetched.Status.Conditions, jobCondition)
- })).Should(Succeed())
-}
-
-func patchVolumeSnapshotStatus(key types.NamespacedName, readyToUse bool) {
- Eventually(testapps.GetAndChangeObjStatus(&testCtx, key, func(fetched *snapshotv1.VolumeSnapshot) {
- snapStatus := snapshotv1.VolumeSnapshotStatus{ReadyToUse: &readyToUse}
- fetched.Status = &snapStatus
- })).Should(Succeed())
-}
-
-func patchBackupPolicySpecBackupStatusUpdates(key types.NamespacedName) {
- Eventually(testapps.GetAndChangeObj(&testCtx, key, func(fetched *dpv1alpha1.BackupPolicy) {
- fetched.Spec.Snapshot.BackupStatusUpdates = []dpv1alpha1.BackupStatusUpdate{
- {
- Path: "manifests.backupLog",
- ContainerName: "postgresql",
- Script: "echo {\"startTime\": \"2023-03-01T00:00:00Z\", \"stopTime\": \"2023-03-01T00:00:00Z\"}",
- UpdateStage: dpv1alpha1.PRE,
- },
- {
- Path: "manifests.backupTool",
- ContainerName: "postgresql",
- Script: "echo {\"FilePath\": \"/backup/test.file\"}",
- UpdateStage: dpv1alpha1.POST,
- },
- }
- })).Should(Succeed())
-}
diff --git a/controllers/dataprotection/backuppolicy_controller.go b/controllers/dataprotection/backuppolicy_controller.go
index a8a26ee7530..1ee9e499e15 100644
--- a/controllers/dataprotection/backuppolicy_controller.go
+++ b/controllers/dataprotection/backuppolicy_controller.go
@@ -21,45 +21,22 @@ package dataprotection
import (
"context"
- "encoding/json"
- "reflect"
- "sort"
- "strings"
- "time"
- "github.com/leaanthony/debme"
- "golang.org/x/exp/slices"
- batchv1 "k8s.io/api/batch/v1"
- corev1 "k8s.io/api/core/v1"
- apierrors "k8s.io/apimachinery/pkg/api/errors"
- "k8s.io/apimachinery/pkg/api/resource"
- metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
- k8sruntime "k8s.io/apimachinery/pkg/runtime"
- "k8s.io/apimachinery/pkg/types"
+ "k8s.io/apimachinery/pkg/runtime"
"k8s.io/client-go/tools/record"
- "k8s.io/client-go/util/workqueue"
ctrl "sigs.k8s.io/controller-runtime"
- "sigs.k8s.io/controller-runtime/pkg/builder"
"sigs.k8s.io/controller-runtime/pkg/client"
- "sigs.k8s.io/controller-runtime/pkg/controller"
- "sigs.k8s.io/controller-runtime/pkg/controller/controllerutil"
- "sigs.k8s.io/controller-runtime/pkg/event"
- "sigs.k8s.io/controller-runtime/pkg/handler"
"sigs.k8s.io/controller-runtime/pkg/log"
- "sigs.k8s.io/controller-runtime/pkg/predicate"
- "sigs.k8s.io/controller-runtime/pkg/source"
- appsv1alpha1 "github.com/apecloud/kubeblocks/apis/apps/v1alpha1"
- dataprotectionv1alpha1 "github.com/apecloud/kubeblocks/apis/dataprotection/v1alpha1"
- "github.com/apecloud/kubeblocks/internal/constant"
+ dpv1alpha1 "github.com/apecloud/kubeblocks/apis/dataprotection/v1alpha1"
intctrlutil "github.com/apecloud/kubeblocks/internal/controllerutil"
- viper "github.com/apecloud/kubeblocks/internal/viperx"
+ dptypes "github.com/apecloud/kubeblocks/internal/dataprotection/types"
)
// BackupPolicyReconciler reconciles a BackupPolicy object
type BackupPolicyReconciler struct {
client.Client
- Scheme *k8sruntime.Scheme
+ Scheme *runtime.Scheme
Recorder record.EventRecorder
}
@@ -67,22 +44,9 @@ type BackupPolicyReconciler struct {
// +kubebuilder:rbac:groups=dataprotection.kubeblocks.io,resources=backuppolicies/status,verbs=get;update;patch
// +kubebuilder:rbac:groups=dataprotection.kubeblocks.io,resources=backuppolicies/finalizers,verbs=update
-// +kubebuilder:rbac:groups=batch,resources=cronjobs,verbs=get;list;watch;create;update;patch;delete
-// +kubebuilder:rbac:groups=batch,resources=cronjobs/status,verbs=get
-// +kubebuilder:rbac:groups=batch,resources=cronjobs/finalizers,verbs=update;patch
-
// Reconcile is part of the main kubernetes reconciliation loop which aims to
-// move the current state of the cluster closer to the desired state.
-// TODO(user): Modify the Reconcile function to compare the state specified by
-// the BackupPolicy object against the actual cluster state, and then
-// perform operations to make the cluster state reflect the state specified by
-// the user.
-//
-// For more details, check Reconcile and its Result here:
-// - https://pkg.go.dev/sigs.k8s.io/controller-runtime@v0.12.1/pkg/reconcile
+// move the current state of the backuppolicy closer to the desired state.
func (r *BackupPolicyReconciler) Reconcile(ctx context.Context, req ctrl.Request) (ctrl.Result, error) {
- // NOTES:
- // setup common request context
reqCtx := intctrlutil.RequestCtx{
Ctx: ctx,
Req: req,
@@ -90,612 +54,51 @@ func (r *BackupPolicyReconciler) Reconcile(ctx context.Context, req ctrl.Request
Recorder: r.Recorder,
}
- backupPolicy := &dataprotectionv1alpha1.BackupPolicy{}
+ backupPolicy := &dpv1alpha1.BackupPolicy{}
if err := r.Client.Get(reqCtx.Ctx, reqCtx.Req.NamespacedName, backupPolicy); err != nil {
return intctrlutil.CheckedRequeueWithError(err, reqCtx.Log, "")
}
- originBackupPolicy := backupPolicy.DeepCopy()
-
// handle finalizer
- res, err := intctrlutil.HandleCRDeletion(reqCtx, r, backupPolicy, dataProtectionFinalizerName, func() (*ctrl.Result, error) {
- return nil, r.deleteExternalResources(reqCtx, backupPolicy)
- })
+ res, err := intctrlutil.HandleCRDeletion(reqCtx, r, backupPolicy, dptypes.DataProtectionFinalizerName,
+ func() (*ctrl.Result, error) {
+ return nil, r.deleteExternalResources(reqCtx, backupPolicy)
+ })
if res != nil {
return *res, err
}
- // try to remove expired or oldest backups, triggered by cronjob controller
- if err = r.removeExpiredBackups(reqCtx); err != nil {
- return r.patchStatusFailed(reqCtx, backupPolicy, "RemoveExpiredBackupsFailed", err)
+ if backupPolicy.Status.ObservedGeneration == backupPolicy.Generation &&
+ backupPolicy.Status.Phase.IsAvailable() {
+ return ctrl.Result{}, nil
}
- if err = r.handleSnapshotPolicy(reqCtx, backupPolicy); err != nil {
- return r.patchStatusFailed(reqCtx, backupPolicy, "HandleSnapshotPolicyFailed", err)
+ patchStatus := func(phase dpv1alpha1.Phase, message string) error {
+ patch := client.MergeFrom(backupPolicy.DeepCopy())
+ backupPolicy.Status.Phase = phase
+ backupPolicy.Status.Message = message
+ backupPolicy.Status.ObservedGeneration = backupPolicy.Generation
+ return r.Status().Patch(ctx, backupPolicy, patch)
}
- if err = r.handleDatafilePolicy(reqCtx, backupPolicy); err != nil {
- return r.patchStatusFailed(reqCtx, backupPolicy, "HandleFullPolicyFailed", err)
- }
+ // TODO(ldm): validate backup policy
- if err = r.handleLogfilePolicy(reqCtx, backupPolicy); err != nil {
- return r.patchStatusFailed(reqCtx, backupPolicy, "HandleIncrementalPolicyFailed", err)
+ if err = patchStatus(dpv1alpha1.AvailablePhase, ""); err != nil {
+ return intctrlutil.CheckedRequeueWithError(err, reqCtx.Log, "")
}
-
- return r.patchStatusAvailable(reqCtx, originBackupPolicy, backupPolicy)
+ intctrlutil.RecordCreatedEvent(r.Recorder, backupPolicy)
+ return ctrl.Result{}, nil
}
// SetupWithManager sets up the controller with the Manager.
func (r *BackupPolicyReconciler) SetupWithManager(mgr ctrl.Manager) error {
return ctrl.NewControllerManagedBy(mgr).
- For(&dataprotectionv1alpha1.BackupPolicy{}).
- Watches(&source.Kind{Type: &dataprotectionv1alpha1.Backup{}}, r.backupDeleteHandler(),
- builder.WithPredicates(predicate.NewPredicateFuncs(filterCreatedByPolicy))).
- WithOptions(controller.Options{
- MaxConcurrentReconciles: viper.GetInt(maxConcurDataProtectionReconKey),
- }).
+ For(&dpv1alpha1.BackupPolicy{}).
Complete(r)
}
-func (r *BackupPolicyReconciler) backupDeleteHandler() *handler.Funcs {
- return &handler.Funcs{
- DeleteFunc: func(event event.DeleteEvent, limitingInterface workqueue.RateLimitingInterface) {
- backup := event.Object.(*dataprotectionv1alpha1.Backup)
- ctx := context.Background()
- backupPolicy := &dataprotectionv1alpha1.BackupPolicy{}
- if err := r.Client.Get(ctx, types.NamespacedName{Name: backup.Spec.BackupPolicyName, Namespace: backup.Namespace}, backupPolicy); err != nil {
- return
- }
- backupType := backup.Spec.BackupType
- // if not refer the backupTool, skip
- commonPolicy := backupPolicy.Spec.GetCommonPolicy(backupType)
- if commonPolicy == nil {
- return
- }
- // if not enable the schedule, skip
- schedulerPolicy := backupPolicy.Spec.GetCommonSchedulePolicy(backupType)
- if schedulerPolicy != nil && !schedulerPolicy.Enable {
- return
- }
- backupTool := &dataprotectionv1alpha1.BackupTool{}
- if err := r.Client.Get(ctx, types.NamespacedName{Name: commonPolicy.BackupToolName}, backupTool); err != nil {
- return
- }
- if backupTool.Spec.DeployKind != dataprotectionv1alpha1.DeployKindStatefulSet {
- return
- }
- _ = r.reconcileForStatefulSetKind(ctx, backupPolicy, backupType, schedulerPolicy.CronExpression)
- },
- }
-}
-
-func (r *BackupPolicyReconciler) deleteExternalResources(reqCtx intctrlutil.RequestCtx, backupPolicy *dataprotectionv1alpha1.BackupPolicy) error {
- // delete cronjob resource
- cronJobList := &batchv1.CronJobList{}
- if err := r.Client.List(reqCtx.Ctx, cronJobList,
- client.InNamespace(viper.GetString(constant.CfgKeyCtrlrMgrNS)),
- client.MatchingLabels{
- dataProtectionLabelBackupPolicyKey: backupPolicy.Name,
- constant.AppManagedByLabelKey: constant.AppName,
- },
- ); err != nil {
- return err
- }
- for _, cronjob := range cronJobList.Items {
- if err := r.removeCronJobFinalizer(reqCtx, &cronjob); err != nil {
- return err
- }
- if err := intctrlutil.BackgroundDeleteObject(r.Client, reqCtx.Ctx, &cronjob); err != nil {
- // failed delete k8s job, return error info.
- return err
- }
- }
- // notice running backup to completed
- backup := &dataprotectionv1alpha1.Backup{}
- for _, v := range []dataprotectionv1alpha1.BackupType{dataprotectionv1alpha1.BackupTypeDataFile,
- dataprotectionv1alpha1.BackupTypeLogFile, dataprotectionv1alpha1.BackupTypeSnapshot} {
- if err := r.Client.Get(reqCtx.Ctx, types.NamespacedName{Namespace: backupPolicy.Namespace,
- Name: getCreatedCRNameByBackupPolicy(backupPolicy, v),
- }, backup); err != nil {
- if apierrors.IsNotFound(err) {
- continue
- }
- return err
- }
- patch := client.MergeFrom(backup.DeepCopy())
- backup.Status.Phase = dataprotectionv1alpha1.BackupCompleted
- backup.Status.CompletionTimestamp = &metav1.Time{Time: time.Now().UTC()}
- if err := r.Client.Status().Patch(reqCtx.Ctx, backup, patch); err != nil {
- return err
- }
- }
- return nil
-}
-
-// patchStatusAvailable patches backup policy status phase to available.
-func (r *BackupPolicyReconciler) patchStatusAvailable(reqCtx intctrlutil.RequestCtx,
- originBackupPolicy,
- backupPolicy *dataprotectionv1alpha1.BackupPolicy) (ctrl.Result, error) {
- if !reflect.DeepEqual(originBackupPolicy.Spec, backupPolicy.Spec) {
- if err := r.Client.Update(reqCtx.Ctx, backupPolicy); err != nil {
- return intctrlutil.CheckedRequeueWithError(err, reqCtx.Log, "")
- }
- }
- // update status phase
- if backupPolicy.Status.Phase != dataprotectionv1alpha1.PolicyAvailable ||
- backupPolicy.Status.ObservedGeneration != backupPolicy.Generation {
- patch := client.MergeFrom(backupPolicy.DeepCopy())
- backupPolicy.Status.ObservedGeneration = backupPolicy.Generation
- backupPolicy.Status.Phase = dataprotectionv1alpha1.PolicyAvailable
- backupPolicy.Status.FailureReason = ""
- if err := r.Client.Status().Patch(reqCtx.Ctx, backupPolicy, patch); err != nil {
- return intctrlutil.CheckedRequeueWithError(err, reqCtx.Log, "")
- }
- }
- return intctrlutil.Reconciled()
-}
-
-// patchStatusFailed patches backup policy status phase to failed.
-func (r *BackupPolicyReconciler) patchStatusFailed(reqCtx intctrlutil.RequestCtx,
- backupPolicy *dataprotectionv1alpha1.BackupPolicy,
- reason string,
- err error) (ctrl.Result, error) {
- if intctrlutil.IsTargetError(err, intctrlutil.ErrorTypeRequeue) {
- return intctrlutil.RequeueAfter(reconcileInterval, reqCtx.Log, "")
- }
- backupPolicyDeepCopy := backupPolicy.DeepCopy()
- backupPolicy.Status.Phase = dataprotectionv1alpha1.PolicyFailed
- backupPolicy.Status.FailureReason = err.Error()
- if !reflect.DeepEqual(backupPolicy.Status, backupPolicyDeepCopy.Status) {
- if patchErr := r.Client.Status().Patch(reqCtx.Ctx, backupPolicy, client.MergeFrom(backupPolicyDeepCopy)); patchErr != nil {
- return intctrlutil.RequeueWithError(patchErr, reqCtx.Log, "")
- }
- }
- r.Recorder.Event(backupPolicy, corev1.EventTypeWarning, reason, err.Error())
- return intctrlutil.RequeueWithError(err, reqCtx.Log, "")
-}
-
-func (r *BackupPolicyReconciler) removeExpiredBackups(reqCtx intctrlutil.RequestCtx) error {
- backups := dataprotectionv1alpha1.BackupList{}
- if err := r.Client.List(reqCtx.Ctx, &backups,
- client.InNamespace(reqCtx.Req.Namespace)); err != nil {
- return err
- }
- now := metav1.Now()
- for _, item := range backups.Items {
- // ignore retained backup.
- if strings.EqualFold(item.GetLabels()[constant.BackupProtectionLabelKey], constant.BackupRetain) {
- continue
- }
- if item.Status.Expiration != nil && item.Status.Expiration.Before(&now) {
- if err := intctrlutil.BackgroundDeleteObject(r.Client, reqCtx.Ctx, &item); err != nil {
- // failed delete backups, return error info.
- return err
- }
- }
- }
- return nil
-}
-
-// removeOldestBackups removes old backups according to backupsHistoryLimit policy.
-func (r *BackupPolicyReconciler) removeOldestBackups(reqCtx intctrlutil.RequestCtx,
- backupPolicyName string,
- backupType dataprotectionv1alpha1.BackupType,
- backupsHistoryLimit int32) error {
- if backupsHistoryLimit == 0 {
- return nil
- }
- matchLabels := map[string]string{
- dataProtectionLabelBackupPolicyKey: backupPolicyName,
- dataProtectionLabelBackupTypeKey: string(backupType),
- dataProtectionLabelAutoBackupKey: "true",
- }
- backups := dataprotectionv1alpha1.BackupList{}
- if err := r.Client.List(reqCtx.Ctx, &backups,
- client.InNamespace(reqCtx.Req.Namespace),
- client.MatchingLabels(matchLabels)); err != nil {
- return err
- }
- // filter final state backups only
- backupItems := []dataprotectionv1alpha1.Backup{}
- for _, item := range backups.Items {
- if item.Status.Phase == dataprotectionv1alpha1.BackupCompleted ||
- item.Status.Phase == dataprotectionv1alpha1.BackupFailed {
- backupItems = append(backupItems, item)
- }
- }
- numToDelete := len(backupItems) - int(backupsHistoryLimit)
- if numToDelete <= 0 {
- return nil
- }
- sort.Sort(byBackupStartTime(backupItems))
- for i := 0; i < numToDelete; i++ {
- if err := intctrlutil.BackgroundDeleteObject(r.Client, reqCtx.Ctx, &backupItems[i]); err != nil {
- // failed delete backups, return error info.
- return err
- }
- }
- return nil
-}
-
-// reconcileForStatefulSetKind reconciles the backup which is controlled by backupPolicy.
-func (r *BackupPolicyReconciler) reconcileForStatefulSetKind(
- ctx context.Context,
- backupPolicy *dataprotectionv1alpha1.BackupPolicy,
- backType dataprotectionv1alpha1.BackupType,
- cronExpression string) error {
- backupName := getCreatedCRNameByBackupPolicy(backupPolicy, backType)
- backup := &dataprotectionv1alpha1.Backup{}
- exists, err := intctrlutil.CheckResourceExists(ctx, r.Client, types.NamespacedName{Name: backupName, Namespace: backupPolicy.Namespace}, backup)
- if err != nil {
- return err
- }
- patch := client.MergeFrom(backup.DeepCopy())
- backup.Name = backupName
- backup.Namespace = backupPolicy.Namespace
- if backup.Labels == nil {
- backup.Labels = map[string]string{}
- }
- backup.Labels[constant.AppManagedByLabelKey] = constant.AppName
- backup.Labels[dataProtectionLabelBackupPolicyKey] = backupPolicy.Name
- backup.Labels[dataProtectionLabelBackupTypeKey] = string(backType)
- backup.Labels[dataProtectionLabelAutoBackupKey] = trueVal
- if !exists {
- if cronExpression == "" {
- return nil
- }
- backup.Spec.BackupType = backType
- backup.Spec.BackupPolicyName = backupPolicy.Name
- return intctrlutil.IgnoreIsAlreadyExists(r.Client.Create(ctx, backup))
- }
-
- // notice to reconcile backup CR
- if cronExpression != "" && slices.Contains([]dataprotectionv1alpha1.BackupPhase{
- dataprotectionv1alpha1.BackupCompleted, dataprotectionv1alpha1.BackupFailed},
- backup.Status.Phase) {
- // if schedule is enabled and backup already is completed, update phase to New
- backup.Status.Phase = dataprotectionv1alpha1.BackupNew
- backup.Status.FailureReason = ""
- return r.Client.Status().Patch(ctx, backup, patch)
- }
- if backup.Annotations == nil {
- backup.Annotations = map[string]string{}
- }
- backup.Annotations[constant.ReconcileAnnotationKey] = time.Now().Format(time.RFC3339Nano)
- return r.Client.Patch(ctx, backup, patch)
-}
-
-// buildCronJob builds cronjob from backup policy.
-func (r *BackupPolicyReconciler) buildCronJob(
- backupPolicy *dataprotectionv1alpha1.BackupPolicy,
- target dataprotectionv1alpha1.TargetCluster,
- cronExpression string,
- backType dataprotectionv1alpha1.BackupType,
- cronJobName string) (*batchv1.CronJob, error) {
- tplFile := "cronjob.cue"
- cueFS, _ := debme.FS(cueTemplates, "cue")
- cueTpl, err := intctrlutil.NewCUETplFromBytes(cueFS.ReadFile(tplFile))
- if err != nil {
- return nil, err
- }
- tolerationPodSpec := corev1.PodSpec{}
- if err = addTolerations(&tolerationPodSpec); err != nil {
- return nil, err
- }
- var ttl metav1.Duration
- if backupPolicy.Spec.Retention != nil && backupPolicy.Spec.Retention.TTL != nil {
- ttl = metav1.Duration{Duration: dataprotectionv1alpha1.ToDuration(backupPolicy.Spec.Retention.TTL)}
- }
- cueValue := intctrlutil.NewCUEBuilder(*cueTpl)
- if cronJobName == "" {
- cronJobName = getCreatedCRNameByBackupPolicy(backupPolicy, backType)
- }
- options := backupPolicyOptions{
- Name: cronJobName,
- BackupPolicyName: backupPolicy.Name,
- Namespace: backupPolicy.Namespace,
- Cluster: target.LabelsSelector.MatchLabels[constant.AppInstanceLabelKey],
- Schedule: cronExpression,
- TTL: ttl,
- BackupType: string(backType),
- ServiceAccount: viper.GetString("KUBEBLOCKS_SERVICEACCOUNT_NAME"),
- MgrNamespace: viper.GetString(constant.CfgKeyCtrlrMgrNS),
- Image: viper.GetString(constant.KBToolsImage),
- Tolerations: &tolerationPodSpec,
- }
- backupPolicyOptionsByte, err := json.Marshal(options)
- if err != nil {
- return nil, err
- }
- if err = cueValue.Fill("options", backupPolicyOptionsByte); err != nil {
- return nil, err
- }
- cuePath := "cronjob"
- if backType == dataprotectionv1alpha1.BackupTypeLogFile {
- cuePath = "cronjob_logfile"
- }
- cronjobByte, err := cueValue.Lookup(cuePath)
- if err != nil {
- return nil, err
- }
-
- cronjob := &batchv1.CronJob{}
- if err = json.Unmarshal(cronjobByte, cronjob); err != nil {
- return nil, err
- }
-
- controllerutil.AddFinalizer(cronjob, dataProtectionFinalizerName)
-
- // set labels
- for k, v := range backupPolicy.Labels {
- if cronjob.Labels == nil {
- cronjob.SetLabels(map[string]string{})
- }
- cronjob.Labels[k] = v
- }
- cronjob.Labels[dataProtectionLabelBackupPolicyKey] = backupPolicy.Name
- cronjob.Labels[dataProtectionLabelBackupTypeKey] = string(backType)
- return cronjob, nil
-}
-
-func (r *BackupPolicyReconciler) removeCronJobFinalizer(reqCtx intctrlutil.RequestCtx, cronjob *batchv1.CronJob) error {
- patch := client.MergeFrom(cronjob.DeepCopy())
- controllerutil.RemoveFinalizer(cronjob, dataProtectionFinalizerName)
- return r.Patch(reqCtx.Ctx, cronjob, patch)
-}
-
-// reconcileCronJob will create/delete/patch cronjob according to cronExpression and policy changes.
-func (r *BackupPolicyReconciler) reconcileCronJob(reqCtx intctrlutil.RequestCtx,
- backupPolicy *dataprotectionv1alpha1.BackupPolicy,
- basePolicy dataprotectionv1alpha1.BasePolicy,
- schedulePolicy *dataprotectionv1alpha1.SchedulePolicy,
- backType dataprotectionv1alpha1.BackupType) error {
- // get cronjob from labels
- cronJob := &batchv1.CronJob{}
- cronJobList := &batchv1.CronJobList{}
- if err := r.Client.List(reqCtx.Ctx, cronJobList,
- client.InNamespace(viper.GetString(constant.CfgKeyCtrlrMgrNS)),
- client.MatchingLabels{
- dataProtectionLabelBackupPolicyKey: backupPolicy.Name,
- dataProtectionLabelBackupTypeKey: string(backType),
- constant.AppManagedByLabelKey: constant.AppName,
- },
- ); err != nil {
- return err
- } else if len(cronJobList.Items) > 0 {
- cronJob = &cronJobList.Items[0]
- }
- if schedulePolicy == nil || !schedulePolicy.Enable {
- if len(cronJob.Name) != 0 {
- // delete the old cronjob.
- if err := r.removeCronJobFinalizer(reqCtx, cronJob); err != nil {
- return err
- }
- return r.Client.Delete(reqCtx.Ctx, cronJob)
- }
- // if no cron expression, return
- return nil
- }
- cronjobProto, err := r.buildCronJob(backupPolicy, basePolicy.Target, schedulePolicy.CronExpression, backType, cronJob.Name)
- if err != nil {
- return err
- }
-
- if backupPolicy.Spec.Schedule.StartingDeadlineMinutes != nil {
- startingDeadlineSeconds := *backupPolicy.Spec.Schedule.StartingDeadlineMinutes * 60
- cronjobProto.Spec.StartingDeadlineSeconds = &startingDeadlineSeconds
- }
- if len(cronJob.Name) == 0 {
- // if no cronjob, create it.
- return r.Client.Create(reqCtx.Ctx, cronjobProto)
- }
- // sync the cronjob with the current backup policy configuration.
- patch := client.MergeFrom(cronJob.DeepCopy())
- cronJob.Spec.StartingDeadlineSeconds = cronjobProto.Spec.StartingDeadlineSeconds
- cronJob.Spec.JobTemplate.Spec.BackoffLimit = &basePolicy.OnFailAttempted
- cronJob.Spec.JobTemplate.Spec.Template = cronjobProto.Spec.JobTemplate.Spec.Template
- cronJob.Spec.Schedule = schedulePolicy.CronExpression
- return r.Client.Patch(reqCtx.Ctx, cronJob, patch)
-}
-
-// handlePolicy handles backup policy.
-func (r *BackupPolicyReconciler) handlePolicy(reqCtx intctrlutil.RequestCtx,
- backupPolicy *dataprotectionv1alpha1.BackupPolicy,
- basePolicy dataprotectionv1alpha1.BasePolicy,
- schedulePolicy *dataprotectionv1alpha1.SchedulePolicy,
- backType dataprotectionv1alpha1.BackupType) error {
-
- if err := r.reconfigure(reqCtx, backupPolicy, basePolicy, backType); err != nil {
- return err
- }
- // create/delete/patch cronjob workload
- if err := r.reconcileCronJob(reqCtx, backupPolicy, basePolicy, schedulePolicy, backType); err != nil {
- return err
- }
- return r.removeOldestBackups(reqCtx, backupPolicy.Name, backType, basePolicy.BackupsHistoryLimit)
-}
-
-// handleSnapshotPolicy handles snapshot policy.
-func (r *BackupPolicyReconciler) handleSnapshotPolicy(
- reqCtx intctrlutil.RequestCtx,
- backupPolicy *dataprotectionv1alpha1.BackupPolicy) error {
- if backupPolicy.Spec.Snapshot == nil {
- // TODO delete cronjob if exists
- return nil
- }
- return r.handlePolicy(reqCtx, backupPolicy, backupPolicy.Spec.Snapshot.BasePolicy,
- backupPolicy.Spec.Schedule.Snapshot, dataprotectionv1alpha1.BackupTypeSnapshot)
-}
-
-// handleDatafilePolicy handles datafile policy.
-func (r *BackupPolicyReconciler) handleDatafilePolicy(
- reqCtx intctrlutil.RequestCtx,
- backupPolicy *dataprotectionv1alpha1.BackupPolicy) error {
- if backupPolicy.Spec.Datafile == nil {
- // TODO delete cronjob if exists
- return nil
- }
- r.setGlobalPersistentVolumeClaim(backupPolicy.Spec.Datafile)
- return r.handlePolicy(reqCtx, backupPolicy, backupPolicy.Spec.Datafile.BasePolicy,
- backupPolicy.Spec.Schedule.Datafile, dataprotectionv1alpha1.BackupTypeDataFile)
-}
-
-// handleLogFilePolicy handles logfile policy.
-func (r *BackupPolicyReconciler) handleLogfilePolicy(
- reqCtx intctrlutil.RequestCtx,
- backupPolicy *dataprotectionv1alpha1.BackupPolicy) error {
- logfile := backupPolicy.Spec.Logfile
- if logfile == nil {
- return nil
- }
- backupTool, err := getBackupToolByName(reqCtx, r.Client, logfile.BackupToolName)
- if err != nil {
- return err
- }
- r.setGlobalPersistentVolumeClaim(logfile)
- schedule := backupPolicy.Spec.Schedule.Logfile
- if backupTool.Spec.DeployKind == dataprotectionv1alpha1.DeployKindStatefulSet {
- var cronExpression string
- if schedule != nil && schedule.Enable {
- cronExpression = schedule.CronExpression
- }
- if err := r.reconfigure(reqCtx, backupPolicy, logfile.BasePolicy, dataprotectionv1alpha1.BackupTypeLogFile); err != nil {
- return err
- }
- return r.reconcileForStatefulSetKind(reqCtx.Ctx, backupPolicy, dataprotectionv1alpha1.BackupTypeLogFile, cronExpression)
- }
- return r.handlePolicy(reqCtx, backupPolicy, logfile.BasePolicy, schedule, dataprotectionv1alpha1.BackupTypeLogFile)
-}
-
-// setGlobalPersistentVolumeClaim sets global config of pvc to common policy.
-func (r *BackupPolicyReconciler) setGlobalPersistentVolumeClaim(backupPolicy *dataprotectionv1alpha1.CommonBackupPolicy) {
- pvcCfg := backupPolicy.PersistentVolumeClaim
- globalPVCName := viper.GetString(constant.CfgKeyBackupPVCName)
- if (pvcCfg.Name == nil || len(*pvcCfg.Name) == 0) && globalPVCName != "" {
- backupPolicy.PersistentVolumeClaim.Name = &globalPVCName
- }
-
- globalInitCapacity := viper.GetString(constant.CfgKeyBackupPVCInitCapacity)
- if pvcCfg.InitCapacity.IsZero() && globalInitCapacity != "" {
- backupPolicy.PersistentVolumeClaim.InitCapacity = resource.MustParse(globalInitCapacity)
- }
-}
-
-type backupReconfigureRef struct {
- Name string `json:"name"`
- Key string `json:"key"`
- Enable parameterPairs `json:"enable,omitempty"`
- Disable parameterPairs `json:"disable,omitempty"`
-}
-
-type parameterPairs map[string][]appsv1alpha1.ParameterPair
-
-func (r *BackupPolicyReconciler) reconfigure(reqCtx intctrlutil.RequestCtx,
- backupPolicy *dataprotectionv1alpha1.BackupPolicy,
- basePolicy dataprotectionv1alpha1.BasePolicy,
- backType dataprotectionv1alpha1.BackupType) error {
-
- reconfigRef := backupPolicy.Annotations[constant.ReconfigureRefAnnotationKey]
- if reconfigRef == "" {
- return nil
- }
- configRef := backupReconfigureRef{}
- if err := json.Unmarshal([]byte(reconfigRef), &configRef); err != nil {
- return err
- }
-
- enable := false
- commonSchedule := backupPolicy.Spec.GetCommonSchedulePolicy(backType)
- if commonSchedule != nil {
- enable = commonSchedule.Enable
- }
- if backupPolicy.Annotations[constant.LastAppliedConfigAnnotationKey] == "" && !enable {
- // disable in the first policy created, no need reconfigure because default configs had been set.
- return nil
- }
- configParameters := configRef.Disable
- if enable {
- configParameters = configRef.Enable
- }
- if configParameters == nil {
- return nil
- }
- parameters := configParameters[string(backType)]
- if len(parameters) == 0 {
- // skip reconfigure if not found parameters.
- return nil
- }
- updateParameterPairsBytes, _ := json.Marshal(parameters)
- updateParameterPairs := string(updateParameterPairsBytes)
- if updateParameterPairs == backupPolicy.Annotations[constant.LastAppliedConfigAnnotationKey] {
- // reconcile the config job if finished
- return r.reconcileReconfigure(reqCtx, backupPolicy)
- }
-
- ops := appsv1alpha1.OpsRequest{
- ObjectMeta: metav1.ObjectMeta{
- GenerateName: backupPolicy.Name + "-",
- Namespace: backupPolicy.Namespace,
- Labels: map[string]string{
- dataProtectionLabelBackupPolicyKey: backupPolicy.Name,
- },
- },
- Spec: appsv1alpha1.OpsRequestSpec{
- Type: appsv1alpha1.ReconfiguringType,
- ClusterRef: basePolicy.Target.LabelsSelector.MatchLabels[constant.AppInstanceLabelKey],
- Reconfigure: &appsv1alpha1.Reconfigure{
- ComponentOps: appsv1alpha1.ComponentOps{
- ComponentName: basePolicy.Target.LabelsSelector.MatchLabels[constant.KBAppComponentLabelKey],
- },
- Configurations: []appsv1alpha1.ConfigurationItem{
- {
- Name: configRef.Name,
- Keys: []appsv1alpha1.ParameterConfig{
- {
- Key: configRef.Key,
- Parameters: parameters,
- },
- },
- },
- },
- },
- },
- }
- if err := r.Client.Create(reqCtx.Ctx, &ops); err != nil {
- return err
- }
-
- r.Recorder.Eventf(backupPolicy, corev1.EventTypeNormal, "Reconfiguring", "update config %s", updateParameterPairs)
- patch := client.MergeFrom(backupPolicy.DeepCopy())
- if backupPolicy.Annotations == nil {
- backupPolicy.Annotations = map[string]string{}
- }
- backupPolicy.Annotations[constant.LastAppliedConfigAnnotationKey] = updateParameterPairs
- if err := r.Client.Patch(reqCtx.Ctx, backupPolicy, patch); err != nil {
- return err
- }
- return intctrlutil.NewErrorf(intctrlutil.ErrorTypeRequeue, "requeue to waiting for ops %s finished.", ops.Name)
-}
-
-func (r *BackupPolicyReconciler) reconcileReconfigure(reqCtx intctrlutil.RequestCtx,
- backupPolicy *dataprotectionv1alpha1.BackupPolicy) error {
-
- opsList := appsv1alpha1.OpsRequestList{}
- if err := r.Client.List(reqCtx.Ctx, &opsList,
- client.InNamespace(backupPolicy.Namespace),
- client.MatchingLabels{dataProtectionLabelBackupPolicyKey: backupPolicy.Name}); err != nil {
- return err
- }
- if len(opsList.Items) > 0 {
- sort.Slice(opsList.Items, func(i, j int) bool {
- return opsList.Items[j].CreationTimestamp.Before(&opsList.Items[i].CreationTimestamp)
- })
- latestOps := opsList.Items[0]
- if latestOps.Status.Phase == appsv1alpha1.OpsFailedPhase {
- return intctrlutil.NewErrorf(intctrlutil.ErrorTypeReconfigureFailed, "ops failed %s", latestOps.Name)
- } else if latestOps.Status.Phase != appsv1alpha1.OpsSucceedPhase {
- return intctrlutil.NewErrorf(intctrlutil.ErrorTypeRequeue, "requeue to waiting for ops %s finished.", latestOps.Name)
- }
- }
+func (r *BackupPolicyReconciler) deleteExternalResources(
+ _ intctrlutil.RequestCtx,
+ _ *dpv1alpha1.BackupPolicy) error {
return nil
}
diff --git a/controllers/dataprotection/backuppolicy_controller_test.go b/controllers/dataprotection/backuppolicy_controller_test.go
index 289f3673660..5a44f8d1266 100644
--- a/controllers/dataprotection/backuppolicy_controller_test.go
+++ b/controllers/dataprotection/backuppolicy_controller_test.go
@@ -20,550 +20,50 @@ along with this program. If not, see .
package dataprotection
import (
- "fmt"
- "time"
-
. "github.com/onsi/ginkgo/v2"
. "github.com/onsi/gomega"
- appsv1 "k8s.io/api/apps/v1"
- batchv1 "k8s.io/api/batch/v1"
- corev1 "k8s.io/api/core/v1"
- metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
- "k8s.io/apimachinery/pkg/types"
"sigs.k8s.io/controller-runtime/pkg/client"
- appsv1alpha1 "github.com/apecloud/kubeblocks/apis/apps/v1alpha1"
- dpv1alpha1 "github.com/apecloud/kubeblocks/apis/dataprotection/v1alpha1"
- "github.com/apecloud/kubeblocks/internal/constant"
intctrlutil "github.com/apecloud/kubeblocks/internal/generics"
testapps "github.com/apecloud/kubeblocks/internal/testutil/apps"
- viper "github.com/apecloud/kubeblocks/internal/viperx"
+ testdp "github.com/apecloud/kubeblocks/internal/testutil/dataprotection"
)
-var _ = Describe("Backup Policy Controller", func() {
- const clusterName = "wesql-cluster"
- const componentName = "replicasets-primary"
- const containerName = "mysql"
- const defaultPVCSize = "1Gi"
- const backupPolicyName = "test-backup-policy"
- const backupRemotePVCName = "backup-remote-pvc"
- const defaultSchedule = "0 3 * * *"
- const defaultTTL = "7d"
- const backupNamePrefix = "test-backup-job-"
- const mgrNamespace = "kube-system"
-
- viper.SetDefault(constant.CfgKeyCtrlrMgrNS, testCtx.DefaultNamespace)
-
+var _ = Describe("BackupPolicy Controller test", func() {
cleanEnv := func() {
// must wait till resources deleted and no longer existed before the testcases start,
// otherwise if later it needs to create some new resource objects with the same name,
// in race conditions, it will find the existence of old objects, resulting failure to
// create the new objects.
By("clean resources")
- viper.SetDefault(constant.CfgKeyCtrlrMgrNS, mgrNamespace)
- // delete rest mocked objects
inNS := client.InNamespace(testCtx.DefaultNamespace)
ml := client.HasLabels{testCtx.TestObjLabelKey}
- // namespaced
- testapps.ClearResources(&testCtx, intctrlutil.ClusterSignature, inNS, ml)
- testapps.ClearResources(&testCtx, intctrlutil.StatefulSetSignature, inNS, ml)
- testapps.ClearResources(&testCtx, intctrlutil.PodSignature, inNS, ml)
- testapps.ClearResources(&testCtx, intctrlutil.BackupPolicySignature, inNS, ml)
- testapps.ClearResourcesWithRemoveFinalizerOption(&testCtx, intctrlutil.BackupSignature, true, inNS)
- testapps.ClearResourcesWithRemoveFinalizerOption(&testCtx, intctrlutil.JobSignature, true, inNS)
- testapps.ClearResources(&testCtx, intctrlutil.CronJobSignature, inNS, ml)
- testapps.ClearResources(&testCtx, intctrlutil.SecretSignature, inNS, ml)
- testapps.ClearResourcesWithRemoveFinalizerOption(&testCtx, intctrlutil.PersistentVolumeClaimSignature, true, inNS)
- // mgr namespaced
- inMgrNS := client.InNamespace(mgrNamespace)
- testapps.ClearResources(&testCtx, intctrlutil.CronJobSignature, inMgrNS, ml)
// non-namespaced
- testapps.ClearResources(&testCtx, intctrlutil.BackupToolSignature, ml)
+ testapps.ClearResourcesWithRemoveFinalizerOption(&testCtx, intctrlutil.ActionSetSignature, true, ml)
+
+ // namespaced
+ testapps.ClearResourcesWithRemoveFinalizerOption(&testCtx, intctrlutil.BackupPolicySignature, true, inNS)
}
BeforeEach(func() {
cleanEnv()
-
- By("By mocking a statefulset")
- sts := testapps.NewStatefulSetFactory(testCtx.DefaultNamespace, clusterName+"-"+componentName, clusterName, componentName).
- AddAppInstanceLabel(clusterName).
- AddContainer(corev1.Container{Name: containerName, Image: testapps.ApeCloudMySQLImage}).
- AddVolumeClaimTemplate(corev1.PersistentVolumeClaim{
- ObjectMeta: metav1.ObjectMeta{Name: testapps.DataVolumeName},
- Spec: testapps.NewPVC(defaultPVCSize),
- }).Create(&testCtx).GetObject()
-
- By("By mocking a pod belonging to the statefulset")
- pod := testapps.NewPodFactory(testCtx.DefaultNamespace, sts.Name+"-0").
- AddAppInstanceLabel(clusterName).
- AddContainer(corev1.Container{Name: containerName, Image: testapps.ApeCloudMySQLImage}).
- Create(&testCtx).GetObject()
-
- By("By mocking a pvc belonging to the pod")
- _ = testapps.NewPersistentVolumeClaimFactory(
- testCtx.DefaultNamespace, "data-"+pod.Name, clusterName, componentName, "data").
- SetStorage("1Gi").
- Create(&testCtx)
})
- AfterEach(cleanEnv)
-
- When("creating backup policy with default settings", func() {
- var backupToolName string
- getCronjobKey := func(backupType dpv1alpha1.BackupType) types.NamespacedName {
- return types.NamespacedName{
- Name: fmt.Sprintf("%s-%s-%s", backupPolicyName, testCtx.DefaultNamespace, backupType),
- Namespace: viper.GetString(constant.CfgKeyCtrlrMgrNS),
- }
- }
-
- BeforeEach(func() {
- viper.Set(constant.CfgKeyCtrlrMgrNS, mgrNamespace)
- viper.Set(constant.CfgKeyCtrlrMgrAffinity,
- "{\"nodeAffinity\":{\"preferredDuringSchedulingIgnoredDuringExecution\":[{\"preference\":{\"matchExpressions\":[{\"key\":\"kb-controller\",\"operator\":\"In\",\"values\":[\"true\"]}]},\"weight\":100}]}}")
- viper.Set(constant.CfgKeyCtrlrMgrTolerations,
- "[{\"key\":\"key1\", \"operator\": \"Exists\", \"effect\": \"NoSchedule\"}]")
- viper.Set(constant.CfgKeyCtrlrMgrNodeSelector, "{\"beta.kubernetes.io/arch\":\"amd64\"}")
-
- By("By creating a backupTool")
- backupTool := testapps.CreateCustomizedObj(&testCtx, "backup/backuptool.yaml",
- &dpv1alpha1.BackupTool{}, testapps.RandomizedObjName())
- backupToolName = backupTool.Name
- })
-
- AfterEach(func() {
- viper.SetDefault(constant.CfgKeyCtrlrMgrNS, testCtx.DefaultNamespace)
- viper.Set(constant.CfgKeyCtrlrMgrAffinity, "")
- viper.Set(constant.CfgKeyCtrlrMgrTolerations, "")
- viper.Set(constant.CfgKeyCtrlrMgrNodeSelector, "")
- })
-
- Context("creates a backup policy", func() {
- var backupPolicyKey types.NamespacedName
- var backupPolicy *dpv1alpha1.BackupPolicy
- var startingDeadlineMinutes int64 = 60
- BeforeEach(func() {
- By("By creating a backupPolicy from backupTool: " + backupToolName)
- backupPolicy = testapps.NewBackupPolicyFactory(testCtx.DefaultNamespace, backupPolicyName).
- AddDataFilePolicy().
- SetBackupToolName(backupToolName).
- SetBackupsHistoryLimit(1).
- SetSchedule(defaultSchedule, true).
- SetScheduleStartingDeadlineMinutes(&startingDeadlineMinutes).
- SetTTL(defaultTTL).
- AddMatchLabels(constant.AppInstanceLabelKey, clusterName).
- SetTargetSecretName(clusterName).
- AddHookPreCommand("touch /data/mysql/.restore;sync").
- AddHookPostCommand("rm -f /data/mysql/.restore;sync").
- SetPVC(backupRemotePVCName).
- Create(&testCtx).GetObject()
- backupPolicyKey = client.ObjectKeyFromObject(backupPolicy)
- })
- It("should success", func() {
- Eventually(testapps.CheckObj(&testCtx, backupPolicyKey, func(g Gomega, fetched *dpv1alpha1.BackupPolicy) {
- g.Expect(fetched.Status.Phase).To(Equal(dpv1alpha1.PolicyAvailable))
- })).Should(Succeed())
- Eventually(testapps.CheckObj(&testCtx, getCronjobKey(dpv1alpha1.BackupTypeDataFile), func(g Gomega, fetched *batchv1.CronJob) {
- g.Expect(fetched.Spec.Schedule).To(Equal(defaultSchedule))
- g.Expect(fetched.Spec.JobTemplate.Spec.Template.Spec.Tolerations).ShouldNot(BeEmpty())
- g.Expect(fetched.Spec.JobTemplate.Spec.Template.Spec.NodeSelector).ShouldNot(BeEmpty())
- g.Expect(fetched.Spec.JobTemplate.Spec.Template.Spec.Affinity).ShouldNot(BeNil())
- g.Expect(fetched.Spec.JobTemplate.Spec.Template.Spec.Affinity.NodeAffinity).ShouldNot(BeNil())
- g.Expect(fetched.Spec.StartingDeadlineSeconds).ShouldNot(BeNil())
- g.Expect(*fetched.Spec.StartingDeadlineSeconds).Should(Equal(startingDeadlineMinutes * 60))
- })).Should(Succeed())
- })
- It("limit backups to 1", func() {
- now := metav1.Now()
- backupStatus := dpv1alpha1.BackupStatus{
- Phase: dpv1alpha1.BackupCompleted,
- Expiration: &now,
- StartTimestamp: &now,
- CompletionTimestamp: &now,
- }
-
- autoBackupLabel := map[string]string{
- dataProtectionLabelAutoBackupKey: "true",
- dataProtectionLabelBackupPolicyKey: backupPolicyName,
- dataProtectionLabelBackupTypeKey: string(dpv1alpha1.BackupTypeDataFile),
- }
-
- By("create a expired backup")
- backupExpired := testapps.NewBackupFactory(testCtx.DefaultNamespace, backupNamePrefix).
- WithRandomName().AddLabelsInMap(autoBackupLabel).
- SetBackupPolicyName(backupPolicyName).
- SetBackupType(dpv1alpha1.BackupTypeDataFile).
- Create(&testCtx).GetObject()
- By("create 1st limit backup")
- backupOutLimit1 := testapps.NewBackupFactory(testCtx.DefaultNamespace, backupNamePrefix).
- WithRandomName().AddLabelsInMap(autoBackupLabel).
- SetBackupPolicyName(backupPolicyName).
- SetBackupType(dpv1alpha1.BackupTypeDataFile).
- Create(&testCtx).GetObject()
- By("create 2nd limit backup")
- backupOutLimit2 := testapps.NewBackupFactory(testCtx.DefaultNamespace, backupNamePrefix).
- WithRandomName().AddLabelsInMap(autoBackupLabel).
- SetBackupPolicyName(backupPolicyName).
- SetBackupType(dpv1alpha1.BackupTypeDataFile).
- Create(&testCtx).GetObject()
-
- By("waiting expired backup completed")
- backupExpiredKey := client.ObjectKeyFromObject(backupExpired)
- patchK8sJobStatus(backupExpiredKey, batchv1.JobComplete)
- Eventually(testapps.CheckObj(&testCtx, backupExpiredKey,
- func(g Gomega, fetched *dpv1alpha1.Backup) {
- g.Expect(fetched.Status.Phase).To(Equal(dpv1alpha1.BackupCompleted))
- })).Should(Succeed())
- By("mock update expired backup status to expire")
- backupStatus.Expiration = &metav1.Time{Time: now.Add(-time.Hour * 24)}
- backupStatus.StartTimestamp = backupStatus.Expiration
- patchBackupStatus(backupStatus, client.ObjectKeyFromObject(backupExpired))
-
- By("waiting 1st limit backup completed")
- backupOutLimit1Key := client.ObjectKeyFromObject(backupOutLimit1)
- patchK8sJobStatus(backupOutLimit1Key, batchv1.JobComplete)
- Eventually(testapps.CheckObj(&testCtx, backupOutLimit1Key,
- func(g Gomega, fetched *dpv1alpha1.Backup) {
- g.Expect(fetched.Status.Phase).To(Equal(dpv1alpha1.BackupCompleted))
- })).Should(Succeed())
- By("mock update 1st limit backup NOT to expire")
- backupStatus.Expiration = &metav1.Time{Time: now.Add(time.Hour * 24)}
- backupStatus.StartTimestamp = &metav1.Time{Time: now.Add(time.Hour)}
- patchBackupStatus(backupStatus, client.ObjectKeyFromObject(backupOutLimit1))
-
- By("waiting 2nd limit backup completed")
- backupOutLimit2Key := client.ObjectKeyFromObject(backupOutLimit2)
- patchK8sJobStatus(backupOutLimit2Key, batchv1.JobComplete)
- Eventually(testapps.CheckObj(&testCtx, backupOutLimit2Key,
- func(g Gomega, fetched *dpv1alpha1.Backup) {
- g.Expect(fetched.Status.Phase).To(Equal(dpv1alpha1.BackupCompleted))
- })).Should(Succeed())
- By("mock update 2nd limit backup NOT to expire")
- backupStatus.Expiration = &metav1.Time{Time: now.Add(time.Hour * 24)}
- backupStatus.StartTimestamp = &metav1.Time{Time: now.Add(time.Hour * 2)}
- patchBackupStatus(backupStatus, client.ObjectKeyFromObject(backupOutLimit2))
-
- // trigger the backup policy controller through update cronjob
- patchCronJobStatus(getCronjobKey(dpv1alpha1.BackupTypeDataFile))
-
- By("retain the latest backup")
- Eventually(testapps.List(&testCtx, intctrlutil.BackupSignature,
- client.MatchingLabels(backupPolicy.Spec.Datafile.Target.LabelsSelector.MatchLabels),
- client.InNamespace(backupPolicy.Namespace))).Should(HaveLen(1))
- })
- })
-
- Context("creates a backup policy with empty schedule", func() {
- var backupPolicyKey types.NamespacedName
- var backupPolicy *dpv1alpha1.BackupPolicy
- BeforeEach(func() {
- By("By creating a backupPolicy from backupTool: " + backupToolName)
- backupPolicy = testapps.NewBackupPolicyFactory(testCtx.DefaultNamespace, backupPolicyName).
- SetBackupToolName(backupToolName).
- AddMatchLabels(constant.AppInstanceLabelKey, clusterName).
- SetTargetSecretName(clusterName).
- AddHookPreCommand("touch /data/mysql/.restore;sync").
- AddHookPostCommand("rm -f /data/mysql/.restore;sync").
- SetPVC(backupRemotePVCName).
- Create(&testCtx).GetObject()
- backupPolicyKey = client.ObjectKeyFromObject(backupPolicy)
- })
- It("should success", func() {
- Eventually(testapps.CheckObj(&testCtx, backupPolicyKey, func(g Gomega, fetched *dpv1alpha1.BackupPolicy) {
- g.Expect(fetched.Status.Phase).To(Equal(dpv1alpha1.PolicyAvailable))
- })).Should(Succeed())
- })
- })
-
- Context("creates a backup policy with invalid schedule", func() {
- var backupPolicyKey types.NamespacedName
- var backupPolicy *dpv1alpha1.BackupPolicy
- BeforeEach(func() {
- By("By creating a backupPolicy from backupTool: " + backupToolName)
- backupPolicy = testapps.NewBackupPolicyFactory(testCtx.DefaultNamespace, backupPolicyName).
- AddSnapshotPolicy().
- SetBackupToolName(backupToolName).
- SetSchedule("invalid schedule", true).
- AddMatchLabels(constant.AppInstanceLabelKey, clusterName).
- SetTargetSecretName(clusterName).
- AddHookPreCommand("touch /data/mysql/.restore;sync").
- AddHookPostCommand("rm -f /data/mysql/.restore;sync").
- SetPVC(backupRemotePVCName).
- Create(&testCtx).GetObject()
- backupPolicyKey = client.ObjectKeyFromObject(backupPolicy)
- })
- It("should failed", func() {
- Eventually(testapps.CheckObj(&testCtx, backupPolicyKey, func(g Gomega, fetched *dpv1alpha1.BackupPolicy) {
- g.Expect(fetched.Status.Phase).NotTo(Equal(dpv1alpha1.PolicyAvailable))
- })).Should(Succeed())
- })
- })
-
- Context("creating a backupPolicy with secret", func() {
- It("creating a backupPolicy with secret", func() {
- By("By creating a backupPolicy with empty secret")
- randomSecretName := testCtx.GetRandomStr()
- backupPolicy := testapps.NewBackupPolicyFactory(testCtx.DefaultNamespace, backupPolicyName).
- AddDataFilePolicy().
- SetBackupToolName(backupToolName).
- AddMatchLabels(constant.AppInstanceLabelKey, clusterName).
- SetTargetSecretName(randomSecretName).
- AddHookPreCommand("touch /data/mysql/.restore;sync").
- AddHookPostCommand("rm -f /data/mysql/.restore;sync").
- SetPVC(backupRemotePVCName).
- Create(&testCtx).GetObject()
- backupPolicyKey := client.ObjectKeyFromObject(backupPolicy)
- Eventually(testapps.CheckObj(&testCtx, backupPolicyKey, func(g Gomega, fetched *dpv1alpha1.BackupPolicy) {
- g.Expect(fetched.Status.Phase).To(Equal(dpv1alpha1.PolicyAvailable))
- g.Expect(fetched.Spec.Datafile.Target.Secret.Name).To(Equal(randomSecretName))
- })).Should(Succeed())
- })
- })
-
- Context("creating a backupPolicy with global backup config", func() {
- It("creating a backupPolicy with global backup config", func() {
- By("By creating a backupPolicy with empty secret")
- pvcName := "backup-data"
- pvcInitCapacity := "10Gi"
- viper.SetDefault(constant.CfgKeyBackupPVCName, pvcName)
- viper.SetDefault(constant.CfgKeyBackupPVCInitCapacity, pvcInitCapacity)
- backupPolicy := testapps.NewBackupPolicyFactory(testCtx.DefaultNamespace, backupPolicyName).
- AddDataFilePolicy().
- SetBackupToolName(backupToolName).
- AddMatchLabels(constant.AppInstanceLabelKey, clusterName).
- AddHookPreCommand("touch /data/mysql/.restore;sync").
- AddHookPostCommand("rm -f /data/mysql/.restore;sync").
- Create(&testCtx).GetObject()
- backupPolicyKey := client.ObjectKeyFromObject(backupPolicy)
- Eventually(testapps.CheckObj(&testCtx, backupPolicyKey, func(g Gomega, fetched *dpv1alpha1.BackupPolicy) {
- g.Expect(fetched.Status.Phase).To(Equal(dpv1alpha1.PolicyAvailable))
- g.Expect(fetched.Spec.Datafile.PersistentVolumeClaim.Name).ToNot(BeNil())
- g.Expect(*fetched.Spec.Datafile.PersistentVolumeClaim.Name).To(Equal(pvcName))
- g.Expect(fetched.Spec.Datafile.PersistentVolumeClaim.InitCapacity.String()).To(Equal(pvcInitCapacity))
- })).Should(Succeed())
- })
- })
- Context("reconcile a logfile backupPolicy", func() {
- It("with reconfigure config and job deployKind", func() {
- By("creating a backupPolicy")
- pvcName := "backup-data"
- pvcInitCapacity := "10Gi"
- viper.SetDefault(constant.CfgKeyBackupPVCName, pvcName)
- viper.SetDefault(constant.CfgKeyBackupPVCInitCapacity, pvcInitCapacity)
- reconfigureRef := `{
- "name": "postgresql-configuration",
- "key": "postgresql.conf",
- "enable": {
- "logfile": [{"key":"archive_command","value":"''"}]
- },
- "disable": {
- "logfile": [{"key": "archive_command","value":"'/bin/true'"}]
- }
- }`
- backupPolicy := testapps.NewBackupPolicyFactory(testCtx.DefaultNamespace, backupPolicyName).
- AddAnnotations(constant.ReconfigureRefAnnotationKey, reconfigureRef).
- AddLogfilePolicy().
- SetBackupToolName(backupToolName).
- AddMatchLabels(constant.AppInstanceLabelKey, clusterName).
- AddSnapshotPolicy().
- AddMatchLabels(constant.AppInstanceLabelKey, clusterName).
- Create(&testCtx).GetObject()
- backupPolicyKey := client.ObjectKeyFromObject(backupPolicy)
- Eventually(testapps.CheckObj(&testCtx, backupPolicyKey, func(g Gomega, fetched *dpv1alpha1.BackupPolicy) {
- g.Expect(fetched.Status.Phase).To(Equal(dpv1alpha1.PolicyAvailable))
- })).Should(Succeed())
- By("enable schedule for reconfigure")
- Eventually(testapps.GetAndChangeObj(&testCtx, backupPolicyKey, func(fetched *dpv1alpha1.BackupPolicy) {
- fetched.Spec.Schedule.Logfile = &dpv1alpha1.SchedulePolicy{Enable: true, CronExpression: "* * * * *"}
- })).Should(Succeed())
- Eventually(testapps.CheckObj(&testCtx, backupPolicyKey, func(g Gomega, fetched *dpv1alpha1.BackupPolicy) {
- g.Expect(fetched.Annotations[constant.LastAppliedConfigAnnotationKey]).To(Equal(`[{"key":"archive_command","value":"''"}]`))
- })).Should(Succeed())
-
- By("disable schedule for reconfigure")
- Eventually(testapps.GetAndChangeObj(&testCtx, backupPolicyKey, func(fetched *dpv1alpha1.BackupPolicy) {
- fetched.Spec.Schedule.Logfile.Enable = false
- })).Should(Succeed())
- Eventually(testapps.CheckObj(&testCtx, backupPolicyKey, func(g Gomega, fetched *dpv1alpha1.BackupPolicy) {
- g.Expect(fetched.Annotations[constant.LastAppliedConfigAnnotationKey]).To(Equal(`[{"key":"archive_command","value":"'/bin/true'"}]`))
- })).Should(Succeed())
- })
-
- It("test logfile backup with a statefulSet deployKind", func() {
-
- // mock a backupTool
- backupTool := createStatefulKindBackupTool()
-
- testLogfileBackupWithStatefulSet := func() {
- By("init test resources")
- // mock a cluster
- cluster := testapps.NewClusterFactory(testCtx.DefaultNamespace, clusterName,
- "test-cd", "test-cv").Create(&testCtx).GetObject()
- // mock a backupPolicy
- backupPolicy := testapps.NewBackupPolicyFactory(testCtx.DefaultNamespace, backupPolicyName).
- SetOwnerReferences("apps.kubeblocks.io/v1alpha1", "Cluster", cluster).
- AddLogfilePolicy().
- SetTTL("7d").
- SetSchedule("*/1 * * * *", false).
- SetBackupToolName(backupTool.Name).
- SetPVC(backupRemotePVCName).
- AddMatchLabels(constant.AppInstanceLabelKey, clusterName).
- Create(&testCtx).GetObject()
-
- By("enable logfile schedule, expect for backup and statefulSet creation")
- Expect(testapps.ChangeObj(&testCtx, backupPolicy, func(policy *dpv1alpha1.BackupPolicy) {
- backupPolicy.Spec.Schedule.Logfile.Enable = true
- })).Should(Succeed())
- backup := &dpv1alpha1.Backup{}
- sts := &appsv1.StatefulSet{}
- backupName := getCreatedCRNameByBackupPolicy(backupPolicy, dpv1alpha1.BackupTypeLogFile)
- Eventually(testapps.CheckObj(&testCtx, types.NamespacedName{
- Name: backupName,
- Namespace: testCtx.DefaultNamespace,
- }, func(g Gomega, tmpBackup *dpv1alpha1.Backup) {
- backup = tmpBackup
- g.Expect(tmpBackup.Status.Phase).Should(Equal(dpv1alpha1.BackupRunning))
- })).Should(Succeed())
- Eventually(testapps.CheckObjExists(&testCtx, types.NamespacedName{
- Name: backupName,
- Namespace: testCtx.DefaultNamespace,
- }, sts, true)).Should(Succeed())
-
- By("check the container envs which is injected successfully.")
- expectedEnv := map[string]string{
- constant.DPArchiveInterval: "60s",
- constant.DPTTL: "7d",
- constant.DPLogfileTTL: "192h",
- constant.DPLogfileTTLSecond: "691200",
- }
- checkGenerateENV := func(sts *appsv1.StatefulSet) {
- mainContainer := sts.Spec.Template.Spec.Containers[0]
- for k, v := range expectedEnv {
- for _, env := range mainContainer.Env {
- if env.Name != k {
- continue
- }
- Expect(env.Value).Should(Equal(v))
- break
- }
- }
- }
- checkGenerateENV(sts)
-
- By("update cronExpression, expect for noticing backup to reconcile")
- Expect(testapps.ChangeObj(&testCtx, backupPolicy, func(policy *dpv1alpha1.BackupPolicy) {
- backupPolicy.Spec.Schedule.Logfile.CronExpression = "*/2 * * * *"
- ttl := "2h"
- backupPolicy.Spec.Retention.TTL = &ttl
- })).Should(Succeed())
- // waiting for sts has changed and expect sts env to change to the corresponding value
- expectedEnv = map[string]string{
- constant.DPArchiveInterval: "120s",
- constant.DPTTL: "2h",
- constant.DPLogfileTTL: "26h",
- constant.DPLogfileTTLSecond: "93600",
- }
- oldStsGeneration := sts.Generation
- Eventually(testapps.CheckObj(&testCtx, types.NamespacedName{
- Name: backupName,
- Namespace: testCtx.DefaultNamespace,
- }, func(g Gomega, tmpSts *appsv1.StatefulSet) {
- g.Expect(tmpSts.Generation).Should(Equal(oldStsGeneration + 1))
- checkGenerateENV(tmpSts)
- })).Should(Succeed())
-
- By("expect to recreate the backup after delete the backup during enable logfile")
- Expect(testapps.ChangeObj(&testCtx, backup, func(policy *dpv1alpha1.Backup) {
- backup.Finalizers = []string{}
- })).Should(Succeed())
- testapps.DeleteObject(&testCtx, client.ObjectKeyFromObject(backup), backup)
- Eventually(testapps.CheckObj(&testCtx, types.NamespacedName{
- Name: backupName,
- Namespace: testCtx.DefaultNamespace,
- }, func(g Gomega, tmpBackup *dpv1alpha1.Backup) {
- g.Expect(tmpBackup.Generation).Should(Equal(int64(1)))
- g.Expect(tmpBackup.Status.Phase).Should(Equal(dpv1alpha1.BackupRunning))
- })).Should(Succeed())
-
- By("disable logfile, expect the backup phase to Completed and sts is deleted")
- Expect(testapps.ChangeObj(&testCtx, backupPolicy, func(policy *dpv1alpha1.BackupPolicy) {
- backupPolicy.Spec.Schedule.Logfile.Enable = false
- })).Should(Succeed())
- Eventually(testapps.CheckObj(&testCtx, types.NamespacedName{
- Name: backupName,
- Namespace: testCtx.DefaultNamespace,
- }, func(g Gomega, tmpBackup *dpv1alpha1.Backup) {
- g.Expect(tmpBackup.Status.Phase).Should(Equal(dpv1alpha1.BackupCompleted))
- })).Should(Succeed())
- Eventually(testapps.CheckObjExists(&testCtx, types.NamespacedName{
- Name: backupName,
- Namespace: testCtx.DefaultNamespace,
- }, sts, false)).Should(Succeed())
-
- By("enable logfile schedule, expect to re-create backup ")
- Expect(testapps.ChangeObj(&testCtx, backupPolicy, func(policy *dpv1alpha1.BackupPolicy) {
- backupPolicy.Spec.Schedule.Logfile.Enable = true
- })).Should(Succeed())
- Eventually(testapps.CheckObj(&testCtx, types.NamespacedName{
- Name: backupName,
- Namespace: testCtx.DefaultNamespace,
- }, func(g Gomega, tmpBackup *dpv1alpha1.Backup) {
- g.Expect(tmpBackup.Status.Phase).Should(Equal(dpv1alpha1.BackupRunning))
- })).Should(Succeed())
-
- By("delete cluster, expect the backup phase to Completed")
- testapps.DeleteObject(&testCtx, types.NamespacedName{
- Name: clusterName,
- Namespace: testCtx.DefaultNamespace,
- }, &appsv1alpha1.Cluster{})
- Eventually(testapps.CheckObj(&testCtx, types.NamespacedName{
- Name: backupName,
- Namespace: testCtx.DefaultNamespace,
- }, func(g Gomega, tmpBackup *dpv1alpha1.Backup) {
- g.Expect(tmpBackup.Status.Phase).Should(Equal(dpv1alpha1.BackupCompleted))
- })).Should(Succeed())
-
- // disabled logfile
- Expect(testapps.ChangeObj(&testCtx, backupPolicy, func(policy *dpv1alpha1.BackupPolicy) {
- backupPolicy.Spec.Schedule.Logfile.Enable = false
- })).Should(Succeed())
- }
-
- testLogfileBackupWithStatefulSet()
-
- // clear backupPolicy
- testapps.ClearResources(&testCtx, intctrlutil.BackupPolicySignature, client.InNamespace(testCtx.DefaultNamespace),
- client.HasLabels{testCtx.TestObjLabelKey})
+ AfterEach(func() {
+ cleanEnv()
+ })
- // test again for create a cluster with same name
- testLogfileBackupWithStatefulSet()
+ Context("create a backup policy", func() {
+ It("backup policy should be available", func() {
+ By("creating actionSet used by backup policy")
+ as := testdp.NewFakeActionSet(&testCtx)
+ Expect(as).ShouldNot(BeNil())
- })
+ By("creating backupPolicy and its status should be available")
+ bp := testdp.NewFakeBackupPolicy(&testCtx, nil)
+ Expect(bp).ShouldNot(BeNil())
})
})
})
-
-func patchBackupStatus(status dpv1alpha1.BackupStatus, key types.NamespacedName) {
- Eventually(testapps.GetAndChangeObjStatus(&testCtx, key, func(fetched *dpv1alpha1.Backup) {
- fetched.Status = status
- })).Should(Succeed())
-}
-
-func patchCronJobStatus(key types.NamespacedName) {
- now := metav1.Now()
- Eventually(testapps.GetAndChangeObjStatus(&testCtx, key, func(fetched *batchv1.CronJob) {
- fetched.Status = batchv1.CronJobStatus{LastSuccessfulTime: &now, LastScheduleTime: &now}
- })).Should(Succeed())
-}
-
-func createStatefulKindBackupTool() *dpv1alpha1.BackupTool {
- By("By creating a backupTool")
- backupTool := testapps.CreateCustomizedObj(&testCtx, "backup/pitr_backuptool.yaml",
- &dpv1alpha1.BackupTool{}, testapps.RandomizedObjName())
- Expect(testapps.ChangeObj(&testCtx, backupTool, func(bt *dpv1alpha1.BackupTool) {
- bt.Spec.DeployKind = dpv1alpha1.DeployKindStatefulSet
- })).Should(Succeed())
- return backupTool
-}
diff --git a/controllers/dataprotection/backuprepo_controller.go b/controllers/dataprotection/backuprepo_controller.go
index e56816e4706..69b547453cb 100644
--- a/controllers/dataprotection/backuprepo_controller.go
+++ b/controllers/dataprotection/backuprepo_controller.go
@@ -24,13 +24,15 @@ import (
"context"
"crypto/md5"
"encoding/hex"
+ "errors"
"fmt"
"reflect"
"sort"
"strings"
"text/template"
- sprig "github.com/go-task/slim-sprig"
+ "github.com/Masterminds/sprig/v3"
+ "github.com/go-logr/logr"
corev1 "k8s.io/api/core/v1"
storagev1 "k8s.io/api/storage/v1"
apierrors "k8s.io/apimachinery/pkg/api/errors"
@@ -46,12 +48,14 @@ import (
"sigs.k8s.io/controller-runtime/pkg/controller/controllerutil"
"sigs.k8s.io/controller-runtime/pkg/handler"
"sigs.k8s.io/controller-runtime/pkg/log"
- "sigs.k8s.io/controller-runtime/pkg/source"
+ "sigs.k8s.io/controller-runtime/pkg/reconcile"
dpv1alpha1 "github.com/apecloud/kubeblocks/apis/dataprotection/v1alpha1"
storagev1alpha1 "github.com/apecloud/kubeblocks/apis/storage/v1alpha1"
"github.com/apecloud/kubeblocks/internal/constant"
intctrlutil "github.com/apecloud/kubeblocks/internal/controllerutil"
+ dptypes "github.com/apecloud/kubeblocks/internal/dataprotection/types"
+ "github.com/apecloud/kubeblocks/internal/generics"
viper "github.com/apecloud/kubeblocks/internal/viperx"
)
@@ -101,11 +105,11 @@ func (r *BackupRepoReconciler) Reconcile(ctx context.Context, req ctrl.Request)
// get repo object
repo := &dpv1alpha1.BackupRepo{}
if err := r.Get(ctx, req.NamespacedName, repo); err != nil {
- return intctrlutil.CheckedRequeueWithError(err, reqCtx.Log, "failed to get BackupRepo")
+ return checkedRequeueWithError(err, reqCtx.Log, "failed to get BackupRepo")
}
// handle finalizer
- res, err := intctrlutil.HandleCRDeletion(reqCtx, r, repo, dataProtectionFinalizerName, func() (*ctrl.Result, error) {
+ res, err := intctrlutil.HandleCRDeletion(reqCtx, r, repo, dptypes.DataProtectionFinalizerName, func() (*ctrl.Result, error) {
return nil, r.deleteExternalResources(reqCtx, repo)
})
if res != nil {
@@ -121,29 +125,48 @@ func (r *BackupRepoReconciler) Reconcile(ctx context.Context, req ctrl.Request)
}
r.providerRefMapper.setRef(repo, types.NamespacedName{Name: repo.Spec.StorageProviderRef})
- // check storage provider status
- provider, err := r.checkStorageProviderStatus(reqCtx, repo)
+ // check storage provider
+ provider, err := r.checkStorageProvider(reqCtx, repo)
if err != nil {
_ = r.updateStatus(reqCtx, repo)
- return intctrlutil.CheckedRequeueWithError(err, reqCtx.Log, "check storage provider status failed")
+ return checkedRequeueWithError(err, reqCtx.Log, "check storage provider status failed")
}
- if !meta.IsStatusConditionTrue(repo.Status.Conditions, ConditionTypeStorageProviderReady) {
- // update status phase to failed
- if err := r.updateStatus(reqCtx, repo); err != nil {
- return intctrlutil.CheckedRequeueWithError(err, reqCtx.Log, "update status phase failed")
- }
- // will reconcile again after the storage provider becomes ready
- return intctrlutil.Reconciled()
+
+ // check parameters for rendering templates
+ parameters, err := r.checkParameters(reqCtx, repo)
+ if err != nil {
+ _ = r.updateStatus(reqCtx, repo)
+ return checkedRequeueWithError(err, reqCtx.Log, "check parameters failed")
+ }
+
+ renderCtx := renderContext{
+ Parameters: parameters,
}
// create StorageClass and Secret for the CSI driver
- err = r.createStorageClassAndSecret(reqCtx, repo, provider)
+ err = r.createStorageClassAndSecret(reqCtx, renderCtx, repo, provider)
if err != nil {
_ = r.updateStatus(reqCtx, repo)
- return intctrlutil.CheckedRequeueWithError(err, reqCtx.Log,
+ return checkedRequeueWithError(err, reqCtx.Log,
"failed to create storage class and secret")
}
+ // check PVC template
+ err = r.checkPVCTemplate(reqCtx, renderCtx, repo, provider)
+ if err != nil {
+ _ = r.updateStatus(reqCtx, repo)
+ return checkedRequeueWithError(err, reqCtx.Log,
+ "failed to check PVC template")
+ }
+
+ // check tool config
+ err = r.checkAndUpdateToolConfig(reqCtx, renderCtx, repo, provider)
+ if err != nil {
+ _ = r.updateStatus(reqCtx, repo)
+ return checkedRequeueWithError(err, reqCtx.Log,
+ "failed to check tool config")
+ }
+
// TODO: implement pre-check logic
// 1. try to create a PVC and observe its status
// 2. create a pre-check job, mount with the PVC and check job status
@@ -151,14 +174,14 @@ func (r *BackupRepoReconciler) Reconcile(ctx context.Context, req ctrl.Request)
// update status phase to ready if all conditions are met
if err = r.updateStatus(reqCtx, repo); err != nil {
- return intctrlutil.CheckedRequeueWithError(err, reqCtx.Log,
+ return checkedRequeueWithError(err, reqCtx.Log,
"failed to update BackupRepo status")
}
// check associated backups, to create PVC in their namespaces
if repo.Status.Phase == dpv1alpha1.BackupRepoReady {
- if err = r.createPVCForAssociatedBackups(reqCtx, repo); err != nil {
- return intctrlutil.CheckedRequeueWithError(err, reqCtx.Log,
+ if err = r.prepareForAssociatedBackups(reqCtx, renderCtx, repo, provider); err != nil {
+ return checkedRequeueWithError(err, reqCtx.Log,
"check associated backups failed")
}
}
@@ -172,12 +195,26 @@ func (r *BackupRepoReconciler) updateStatus(reqCtx intctrlutil.RequestCtx, repo
if repo.Status.Phase != dpv1alpha1.BackupRepoDeleting {
phase := dpv1alpha1.BackupRepoFailed
if meta.IsStatusConditionTrue(repo.Status.Conditions, ConditionTypeStorageProviderReady) &&
- meta.IsStatusConditionTrue(repo.Status.Conditions, ConditionTypeStorageClassCreated) {
+ meta.IsStatusConditionTrue(repo.Status.Conditions, ConditionTypeParametersChecked) &&
+ meta.IsStatusConditionTrue(repo.Status.Conditions, ConditionTypeStorageClassCreated) &&
+ meta.IsStatusConditionTrue(repo.Status.Conditions, ConditionTypePVCTemplateChecked) &&
+ meta.IsStatusConditionTrue(repo.Status.Conditions, ConditionTypeToolConfigChecked) {
phase = dpv1alpha1.BackupRepoReady
}
repo.Status.Phase = phase
}
- repo.Status.IsDefault = repo.Annotations[constant.DefaultBackupRepoAnnotationKey] == trueVal
+ repo.Status.IsDefault = repo.Annotations[dptypes.DefaultBackupRepoAnnotationKey] == trueVal
+
+ // update other fields
+ if repo.Status.BackupPVCName == "" {
+ repo.Status.BackupPVCName = randomNameForDerivedObject(repo, "pvc")
+ }
+ if repo.Status.ToolConfigSecretName == "" {
+ repo.Status.ToolConfigSecretName = randomNameForDerivedObject(repo, "tool-config")
+ }
+ if repo.Status.ObservedGeneration != repo.Generation {
+ repo.Status.ObservedGeneration = repo.Generation
+ }
if !reflect.DeepEqual(old.Status, repo.Status) {
if err := r.Client.Status().Patch(reqCtx.Ctx, repo, client.MergeFrom(old)); err != nil {
@@ -187,60 +224,95 @@ func (r *BackupRepoReconciler) updateStatus(reqCtx intctrlutil.RequestCtx, repo
return nil
}
-func (r *BackupRepoReconciler) checkStorageProviderStatus(
- reqCtx intctrlutil.RequestCtx, repo *dpv1alpha1.BackupRepo) (*storagev1alpha1.StorageProvider, error) {
- var condType = ConditionTypeStorageProviderReady
- var status metav1.ConditionStatus
- var reason string
- var message string
+func (r *BackupRepoReconciler) updateConditionInDefer(reqCtx intctrlutil.RequestCtx, repo *dpv1alpha1.BackupRepo,
+ condType string, reason string, err *error) {
+ status := metav1.ConditionTrue
+ message := ""
+ if *err != nil {
+ status = metav1.ConditionFalse
+ message = (*err).Error()
+ }
+ updateErr := updateCondition(reqCtx.Ctx, r.Client, repo, condType, status, reason, message)
+ if *err == nil {
+ *err = updateErr
+ }
+}
+
+func (r *BackupRepoReconciler) checkStorageProvider(
+ reqCtx intctrlutil.RequestCtx, repo *dpv1alpha1.BackupRepo) (provider *storagev1alpha1.StorageProvider, err error) {
+ reason := ReasonUnknownError
+ defer func() {
+ r.updateConditionInDefer(reqCtx, repo, ConditionTypeStorageProviderReady, reason, &err)
+ }()
// get storage provider object
providerKey := client.ObjectKey{Name: repo.Spec.StorageProviderRef}
- provider := &storagev1alpha1.StorageProvider{}
- err := r.Client.Get(reqCtx.Ctx, providerKey, provider)
+ provider = &storagev1alpha1.StorageProvider{}
+ err = r.Client.Get(reqCtx.Ctx, providerKey, provider)
if err != nil {
if apierrors.IsNotFound(err) {
- status = metav1.ConditionFalse
reason = ReasonStorageProviderNotFound
- } else {
- status = metav1.ConditionUnknown
- reason = ReasonUnknownError
- message = err.Error()
}
- _ = updateCondition(reqCtx.Ctx, r.Client, repo, condType, status, reason, message)
return nil, err
}
+ // check its spec
+ switch {
+ case repo.AccessByMount():
+ if provider.Spec.StorageClassTemplate == "" &&
+ provider.Spec.PersistentVolumeClaimTemplate == "" {
+ // both StorageClassTemplate and PersistentVolumeClaimTemplate are empty.
+ // in this case, we are unable to create a backup PVC.
+ reason = ReasonInvalidStorageProvider
+ return provider, newDependencyError("both StorageClassTemplate and PersistentVolumeClaimTemplate are empty")
+ }
+ case repo.AccessByTool():
+ if provider.Spec.DatasafedConfigTemplate == "" {
+ reason = ReasonInvalidStorageProvider
+ return provider, newDependencyError("DatasafedConfigTemplate is empty")
+ }
+ }
+
// check its status
if provider.Status.Phase == storagev1alpha1.StorageProviderReady {
- status = metav1.ConditionTrue
reason = ReasonStorageProviderReady
+ return provider, nil
} else {
- status = metav1.ConditionFalse
reason = ReasonStorageProviderNotReady
- message = fmt.Sprintf("storage provider %s is not ready, status: %s",
- provider.Name, provider.Status.Phase)
- }
- if updateErr := updateCondition(reqCtx.Ctx, r.Client, repo, condType, status, reason, message); updateErr != nil {
- return nil, updateErr
+ err = newDependencyError(fmt.Sprintf("storage provider %s is not ready, status: %s",
+ provider.Name, provider.Status.Phase))
+ return provider, err
}
- return provider, nil
}
-func (r *BackupRepoReconciler) createStorageClassAndSecret(
- reqCtx intctrlutil.RequestCtx, repo *dpv1alpha1.BackupRepo, provider *storagev1alpha1.StorageProvider) error {
+func (r *BackupRepoReconciler) checkParameters(reqCtx intctrlutil.RequestCtx,
+ repo *dpv1alpha1.BackupRepo) (parameters map[string]string, err error) {
+ reason := ReasonUnknownError
+ defer func() {
+ r.updateConditionInDefer(reqCtx, repo, ConditionTypeParametersChecked, reason, &err)
+ }()
// collect parameters for rendering templates
- parameters, err := r.collectParameters(reqCtx, repo)
+ parameters, err = r.collectParameters(reqCtx, repo)
if err != nil {
- _ = updateCondition(reqCtx.Ctx, r.Client, repo, ConditionTypeStorageClassCreated,
- metav1.ConditionUnknown, ReasonUnknownError, err.Error())
- return fmt.Errorf("failed to collect render parameters: %w", err)
+ if apierrors.IsNotFound(err) {
+ reason = ReasonCredentialSecretNotFound
+ }
+ return nil, err
}
// TODO: verify parameters
- renderCtx := renderContext{
- Parameters: parameters,
- }
+ reason = ReasonParametersChecked
+ return parameters, nil
+}
+
+func (r *BackupRepoReconciler) createStorageClassAndSecret(reqCtx intctrlutil.RequestCtx,
+ renderCtx renderContext, repo *dpv1alpha1.BackupRepo, provider *storagev1alpha1.StorageProvider) (err error) {
+
+ reason := ReasonUnknownError
+ defer func() {
+ r.updateConditionInDefer(reqCtx, repo, ConditionTypeStorageClassCreated, reason, &err)
+ }()
+
oldRepo := repo.DeepCopy()
// create secret for the CSI driver if it's not exist,
@@ -253,27 +325,24 @@ func (r *BackupRepoReconciler) createStorageClassAndSecret(
}
}
renderCtx.CSIDriverSecretRef = *repo.Status.GeneratedCSIDriverSecret
- // create secret if it's not exist
- if _, err := r.createSecretForCSIDriver(reqCtx, renderCtx, repo, provider); err != nil {
+ // create or update the secret for CSI
+ if _, err = r.createOrUpdateSecretForCSIDriver(reqCtx, renderCtx, repo, provider); err != nil {
+ reason = ReasonPrepareCSISecretFailed
return err
}
}
- // create storage class if it's not exist
- if repo.Status.GeneratedStorageClassName == "" {
- repo.Status.GeneratedStorageClassName = randomNameForDerivedObject(repo, "sc")
- }
- if _, err := r.createStorageClass(reqCtx, renderCtx, repo, provider); err != nil {
- return err
+ if provider.Spec.StorageClassTemplate != "" {
+ // create storage class if it's not exist
+ if repo.Status.GeneratedStorageClassName == "" {
+ repo.Status.GeneratedStorageClassName = randomNameForDerivedObject(repo, "sc")
+ }
+ if _, err = r.createStorageClass(reqCtx, renderCtx, repo, provider); err != nil {
+ reason = ReasonPrepareStorageClassFailed
+ return err
+ }
}
- // update other fields
- if repo.Status.BackupPVCName == "" {
- repo.Status.BackupPVCName = randomNameForDerivedObject(repo, "pvc")
- }
- if repo.Status.ObservedGeneration != repo.Generation {
- repo.Status.ObservedGeneration = repo.Generation
- }
if !meta.IsStatusConditionTrue(repo.Status.Conditions, ConditionTypeStorageClassCreated) {
setCondition(repo, ConditionTypeStorageClassCreated,
metav1.ConditionTrue, ReasonStorageClassCreated, "")
@@ -285,28 +354,34 @@ func (r *BackupRepoReconciler) createStorageClassAndSecret(
return fmt.Errorf("failed to patch backup repo: %w", err)
}
}
+ reason = ReasonStorageClassCreated
return nil
}
-func (r *BackupRepoReconciler) createSecretForCSIDriver(
+func (r *BackupRepoReconciler) createOrUpdateSecretForCSIDriver(
reqCtx intctrlutil.RequestCtx, renderCtx renderContext,
repo *dpv1alpha1.BackupRepo, provider *storagev1alpha1.StorageProvider) (created bool, err error) {
- secretTemplateMD5 := md5Digest(provider.Spec.CSIDriverSecretTemplate)
- templateValuesMD5 := md5Digest(stableSerializeMap(renderCtx.Parameters))
- condType := ConditionTypeStorageClassCreated
- setSecretContent := func(secret *corev1.Secret) error {
+ secret := &corev1.Secret{}
+ secret.Name = repo.Status.GeneratedCSIDriverSecret.Name
+ secret.Namespace = repo.Status.GeneratedCSIDriverSecret.Namespace
+
+ templateMd5 := md5Digest(provider.Spec.CSIDriverSecretTemplate)
+ parametersMd5 := renderCtx.Md5OfParameters()
+ shouldUpdateFunc := func() bool {
+ tmplMd5InSecret := secret.Annotations[dataProtectionSecretTemplateMD5AnnotationKey]
+ paramMd5InSecret := secret.Annotations[dataProtectionTemplateValuesMD5AnnotationKey]
+ return templateMd5 != tmplMd5InSecret || parametersMd5 != paramMd5InSecret
+ }
+
+ return createOrUpdateObject(reqCtx.Ctx, r.Client, secret, func() error {
// render secret template
content, err := renderTemplate("secret", provider.Spec.CSIDriverSecretTemplate, renderCtx)
if err != nil {
- _ = updateCondition(reqCtx.Ctx, r.Client, repo, condType,
- metav1.ConditionFalse, ReasonBadSecretTemplate, err.Error())
return fmt.Errorf("failed to render secret template: %w", err)
}
secretStringData := map[string]string{}
if err = yaml.Unmarshal([]byte(content), &secretStringData); err != nil {
- _ = updateCondition(reqCtx.Ctx, r.Client, repo, condType,
- metav1.ConditionFalse, ReasonBadSecretTemplate, err.Error())
return fmt.Errorf("failed to unmarshal secret content: %w", err)
}
secretData := make(map[string][]byte, len(secretStringData))
@@ -314,63 +389,24 @@ func (r *BackupRepoReconciler) createSecretForCSIDriver(
secretData[k] = []byte(v)
}
secret.Data = secretData
- return nil
- }
-
- secret := &corev1.Secret{}
- secret.Name = repo.Status.GeneratedCSIDriverSecret.Name
- secret.Namespace = repo.Status.GeneratedCSIDriverSecret.Namespace
-
- // create the secret object if not exist.
- // this function will retrieve the whole secret object
- // when the object is existing.
- created, err = createObjectIfNotExist(reqCtx.Ctx, r.Client, secret,
- func() error {
- secret.Labels = map[string]string{
- dataProtectionBackupRepoKey: repo.Name,
- }
- secret.Annotations = map[string]string{
- dataProtectionSecretTemplateMD5AnnotationKey: secretTemplateMD5,
- dataProtectionTemplateValuesMD5AnnotationKey: templateValuesMD5,
- }
- if err := setSecretContent(secret); err != nil {
- return err
- }
- if err := controllerutil.SetControllerReference(repo, secret, r.Scheme); err != nil {
- _ = updateCondition(reqCtx.Ctx, r.Client, repo, condType,
- metav1.ConditionUnknown, ReasonUnknownError, err.Error())
- return fmt.Errorf("failed to set controller reference: %w", err)
- }
- return nil
- })
- if err != nil {
- return false, fmt.Errorf("createObjectIfNotExist for secret %s failed: %w",
- client.ObjectKeyFromObject(secret), err)
- }
- if created {
- return true, nil
- }
- // check if the template or config changed, then update the secret
- currSecretTemplateMD5 := secret.Annotations[dataProtectionSecretTemplateMD5AnnotationKey]
- currTemplateValuesMD5 := secret.Annotations[dataProtectionTemplateValuesMD5AnnotationKey]
- if currSecretTemplateMD5 != secretTemplateMD5 || currTemplateValuesMD5 != templateValuesMD5 {
- patch := client.MergeFrom(secret.DeepCopy())
- if err := setSecretContent(secret); err != nil {
- return false, err
+ // set labels and annotations
+ if secret.Labels == nil {
+ secret.Labels = make(map[string]string)
}
+ secret.Labels[dataProtectionBackupRepoKey] = repo.Name
+
if secret.Annotations == nil {
secret.Annotations = make(map[string]string)
}
- secret.Annotations[dataProtectionSecretTemplateMD5AnnotationKey] = secretTemplateMD5
- secret.Annotations[dataProtectionTemplateValuesMD5AnnotationKey] = templateValuesMD5
- err := r.Client.Patch(reqCtx.Ctx, secret, patch)
- if err != nil {
- return false, fmt.Errorf("failed to patch secret object %s: %w",
- client.ObjectKeyFromObject(secret), err)
+ secret.Annotations[dataProtectionSecretTemplateMD5AnnotationKey] = templateMd5
+ secret.Annotations[dataProtectionTemplateValuesMD5AnnotationKey] = parametersMd5
+
+ if err := controllerutil.SetControllerReference(repo, secret, r.Scheme); err != nil {
+ return fmt.Errorf("failed to set controller reference: %w", err)
}
- }
- return false, nil
+ return nil
+ }, shouldUpdateFunc)
}
func (r *BackupRepoReconciler) createStorageClass(
@@ -381,18 +417,12 @@ func (r *BackupRepoReconciler) createStorageClass(
storageClass.Name = repo.Status.GeneratedStorageClassName
return createObjectIfNotExist(reqCtx.Ctx, r.Client, storageClass,
func() error {
- condType := ConditionTypeStorageClassCreated
-
// render storage class template
content, err := renderTemplate("sc", provider.Spec.StorageClassTemplate, renderCtx)
if err != nil {
- _ = updateCondition(reqCtx.Ctx, r.Client, repo, condType,
- metav1.ConditionFalse, ReasonBadStorageClassTemplate, err.Error())
return fmt.Errorf("failed to render storage class template: %w", err)
}
if err = yaml.Unmarshal([]byte(content), storageClass); err != nil {
- _ = updateCondition(reqCtx.Ctx, r.Client, repo, condType,
- metav1.ConditionFalse, ReasonBadStorageClassTemplate, err.Error())
return fmt.Errorf("failed to unmarshal storage class: %w", err)
}
@@ -406,14 +436,123 @@ func (r *BackupRepoReconciler) createStorageClass(
storageClass.ReclaimPolicy = &repo.Spec.PVReclaimPolicy
}
if err := controllerutil.SetControllerReference(repo, storageClass, r.Scheme); err != nil {
- _ = updateCondition(reqCtx.Ctx, r.Client, repo, condType,
- metav1.ConditionUnknown, ReasonUnknownError, err.Error())
return fmt.Errorf("failed to set owner reference: %w", err)
}
return nil
})
}
+func (r *BackupRepoReconciler) checkPVCTemplate(reqCtx intctrlutil.RequestCtx,
+ renderCtx renderContext, repo *dpv1alpha1.BackupRepo, provider *storagev1alpha1.StorageProvider) (err error) {
+
+ reason := ReasonUnknownError
+ defer func() {
+ r.updateConditionInDefer(reqCtx, repo, ConditionTypePVCTemplateChecked, reason, &err)
+ }()
+
+ if !repo.AccessByMount() || provider.Spec.PersistentVolumeClaimTemplate == "" {
+ return nil
+ }
+ checkedTemplateMd5 := repo.Annotations[dataProtectionPVCTemplateMD5MD5AnnotationKey]
+ checkedParametersMd5 := repo.Annotations[dataProtectionTemplateValuesMD5AnnotationKey]
+ currentTemplateMd5 := md5Digest(provider.Spec.PersistentVolumeClaimTemplate)
+ currentParametersMd5 := renderCtx.Md5OfParameters()
+ if checkedTemplateMd5 != currentTemplateMd5 || checkedParametersMd5 != currentParametersMd5 {
+ pvc := &corev1.PersistentVolumeClaim{}
+ err := r.constructPVCByTemplate(renderCtx, pvc, repo, provider.Spec.PersistentVolumeClaimTemplate)
+ if err != nil {
+ reason = ReasonBadPVCTemplate
+ return err
+ }
+ }
+ if err = updateAnnotations(reqCtx.Ctx, r.Client, repo, map[string]string{
+ dataProtectionPVCTemplateMD5MD5AnnotationKey: currentTemplateMd5,
+ dataProtectionTemplateValuesMD5AnnotationKey: currentParametersMd5,
+ }); err != nil {
+ return err
+ }
+ reason = ReasonPVCTemplateChecked
+ return nil
+}
+
+func (r *BackupRepoReconciler) checkAndUpdateToolConfig(reqCtx intctrlutil.RequestCtx,
+ renderCtx renderContext, repo *dpv1alpha1.BackupRepo, provider *storagev1alpha1.StorageProvider) (err error) {
+
+ reason := ReasonUnknownError
+ defer func() {
+ r.updateConditionInDefer(reqCtx, repo, ConditionTypeToolConfigChecked, reason, &err)
+ }()
+
+ if !repo.AccessByTool() {
+ return nil
+ }
+ checkedTemplateMd5 := repo.Annotations[dataProtectionToolConfigTemplateMD5MD5AnnotationKey]
+ checkedParametersMd5 := repo.Annotations[dataProtectionTemplateValuesMD5AnnotationKey]
+ currentTemplateMd5 := md5Digest(provider.Spec.DatasafedConfigTemplate)
+ currentParametersMd5 := renderCtx.Md5OfParameters()
+ if !(checkedTemplateMd5 != currentTemplateMd5 || checkedParametersMd5 != currentParametersMd5) {
+ return nil
+ }
+ // check tool config template
+ content, err := renderTemplate("tool-config", provider.Spec.DatasafedConfigTemplate, renderCtx)
+ if err != nil {
+ reason = ReasonBadToolConfigTemplate
+ return err
+ }
+ // update existing tool config secrets
+ secretList := &corev1.SecretList{}
+ err = r.Client.List(reqCtx.Ctx, secretList, client.MatchingLabels{
+ dataProtectionBackupRepoKey: repo.Name,
+ dataProtectionIsToolConfigKey: trueVal,
+ })
+ if err != nil {
+ return err
+ }
+ for idx := range secretList.Items {
+ secret := &secretList.Items[idx]
+ tmplMd5InSecret := secret.Annotations[dataProtectionToolConfigTemplateMD5MD5AnnotationKey]
+ paramMd5InSecret := secret.Annotations[dataProtectionTemplateValuesMD5AnnotationKey]
+ if tmplMd5InSecret == currentTemplateMd5 && paramMd5InSecret == currentParametersMd5 {
+ continue
+ }
+ patch := client.MergeFrom(secret.DeepCopy())
+ constructToolConfigSecret(secret, content)
+ if secret.Annotations == nil {
+ secret.Annotations = make(map[string]string)
+ }
+ secret.Annotations[dataProtectionToolConfigTemplateMD5MD5AnnotationKey] = currentTemplateMd5
+ secret.Annotations[dataProtectionTemplateValuesMD5AnnotationKey] = currentParametersMd5
+ if err = r.Client.Patch(reqCtx.Ctx, secret, patch); err != nil {
+ return err
+ }
+ }
+
+ if err = updateAnnotations(reqCtx.Ctx, r.Client, repo, map[string]string{
+ dataProtectionToolConfigTemplateMD5MD5AnnotationKey: currentTemplateMd5,
+ dataProtectionTemplateValuesMD5AnnotationKey: currentParametersMd5,
+ }); err != nil {
+ return err
+ }
+ reason = ReasonToolConfigChecked
+ return nil
+}
+
+func (r *BackupRepoReconciler) constructPVCByTemplate(
+ renderCtx renderContext, pvc *corev1.PersistentVolumeClaim,
+ repo *dpv1alpha1.BackupRepo, tmpl string) error {
+ // fill render values
+ renderCtx.GeneratedStorageClassName = repo.Status.GeneratedStorageClassName
+
+ content, err := renderTemplate("pvc", tmpl, renderCtx)
+ if err != nil {
+ return fmt.Errorf("failed to render PVC template: %w", err)
+ }
+ if err = yaml.Unmarshal([]byte(content), pvc); err != nil {
+ return fmt.Errorf("failed to unmarshal PVC object: %w", err)
+ }
+ return nil
+}
+
func (r *BackupRepoReconciler) listAssociatedBackups(
reqCtx intctrlutil.RequestCtx, repo *dpv1alpha1.BackupRepo, extraSelector map[string]string) ([]*dpv1alpha1.Backup, error) {
// list backups associated with the repo
@@ -428,7 +567,7 @@ func (r *BackupRepoReconciler) listAssociatedBackups(
var filtered []*dpv1alpha1.Backup
for idx := range backupList.Items {
backup := &backupList.Items[idx]
- if backup.Status.Phase == dpv1alpha1.BackupFailed {
+ if backup.Status.Phase == dpv1alpha1.BackupPhaseFailed {
continue
}
filtered = append(filtered, backup)
@@ -436,10 +575,12 @@ func (r *BackupRepoReconciler) listAssociatedBackups(
return filtered, err
}
-func (r *BackupRepoReconciler) createPVCForAssociatedBackups(
- reqCtx intctrlutil.RequestCtx, repo *dpv1alpha1.BackupRepo) error {
+func (r *BackupRepoReconciler) prepareForAssociatedBackups(
+ reqCtx intctrlutil.RequestCtx, renderCtx renderContext,
+ repo *dpv1alpha1.BackupRepo, provider *storagev1alpha1.StorageProvider) error {
+
backups, err := r.listAssociatedBackups(reqCtx, repo, map[string]string{
- dataProtectionNeedRepoPVCKey: trueVal,
+ dataProtectionWaitRepoPreparationKey: trueVal,
})
if err != nil {
return err
@@ -447,14 +588,26 @@ func (r *BackupRepoReconciler) createPVCForAssociatedBackups(
// return any error to reconcile the repo
var retErr error
for _, backup := range backups {
- if err := r.checkOrCreatePVC(reqCtx, repo, backup.Namespace); err != nil {
- reqCtx.Log.Error(err, "failed to check or create PVC", "namespace", backup.Namespace)
- retErr = err
- continue
+ switch {
+ case repo.AccessByMount():
+ if err := r.checkOrCreatePVC(reqCtx, renderCtx, repo, provider, backup.Namespace); err != nil {
+ reqCtx.Log.Error(err, "failed to check or create PVC", "namespace", backup.Namespace)
+ retErr = err
+ continue
+ }
+ case repo.AccessByTool():
+ if err := r.checkOrCreateToolConfigSecret(reqCtx, renderCtx, repo, provider, backup.Namespace); err != nil {
+ reqCtx.Log.Error(err, "failed to check or create tool config secret", "namespace", backup.Namespace)
+ retErr = err
+ continue
+ }
+ default:
+ retErr = fmt.Errorf("unknown access method: %s", repo.Spec.AccessMethod)
}
- if backup.Labels[dataProtectionNeedRepoPVCKey] != "" {
+
+ if backup.Labels[dataProtectionWaitRepoPreparationKey] != "" {
patch := client.MergeFrom(backup.DeepCopy())
- delete(backup.Labels, dataProtectionNeedRepoPVCKey)
+ delete(backup.Labels, dataProtectionWaitRepoPreparationKey)
if err = r.Client.Patch(reqCtx.Ctx, backup, patch); err != nil {
reqCtx.Log.Error(err, "failed to patch backup",
"backup", client.ObjectKeyFromObject(backup))
@@ -467,30 +620,49 @@ func (r *BackupRepoReconciler) createPVCForAssociatedBackups(
}
func (r *BackupRepoReconciler) checkOrCreatePVC(
- reqCtx intctrlutil.RequestCtx, repo *dpv1alpha1.BackupRepo, namespace string) error {
+ reqCtx intctrlutil.RequestCtx, renderCtx renderContext,
+ repo *dpv1alpha1.BackupRepo, provider *storagev1alpha1.StorageProvider, namespace string) error {
+
pvc := &corev1.PersistentVolumeClaim{}
pvc.Name = repo.Status.BackupPVCName
pvc.Namespace = namespace
_, err := createObjectIfNotExist(reqCtx.Ctx, r.Client, pvc,
func() error {
- storageClassName := repo.Status.GeneratedStorageClassName
- volumeMode := corev1.PersistentVolumeFilesystem
- resources := corev1.ResourceRequirements{}
- if !repo.Spec.VolumeCapacity.IsZero() {
- resources.Requests = corev1.ResourceList{
- corev1.ResourceStorage: repo.Spec.VolumeCapacity,
+ if provider.Spec.PersistentVolumeClaimTemplate != "" {
+ // construct the PVC object by rendering the template
+ err := r.constructPVCByTemplate(renderCtx, pvc, repo, provider.Spec.PersistentVolumeClaimTemplate)
+ if err != nil {
+ return err
+ }
+ // overwrite PVC name and namespace
+ pvc.Name = repo.Status.BackupPVCName
+ pvc.Namespace = namespace
+ } else {
+ // set storage class name to PVC, other fields will be set with default value later
+ storageClassName := repo.Status.GeneratedStorageClassName
+ pvc.Spec = corev1.PersistentVolumeClaimSpec{
+ StorageClassName: &storageClassName,
}
}
- pvc.Labels = map[string]string{
- dataProtectionBackupRepoKey: repo.Name,
+ // add a referencing label
+ if pvc.Labels == nil {
+ pvc.Labels = make(map[string]string)
}
- pvc.Spec = corev1.PersistentVolumeClaimSpec{
- AccessModes: []corev1.PersistentVolumeAccessMode{
- corev1.ReadWriteMany,
- },
- Resources: resources,
- StorageClassName: &storageClassName,
- VolumeMode: &volumeMode,
+ pvc.Labels[dataProtectionBackupRepoKey] = repo.Name
+ // set default values if not set
+ if len(pvc.Spec.AccessModes) == 0 {
+ pvc.Spec.AccessModes = []corev1.PersistentVolumeAccessMode{corev1.ReadWriteMany}
+ }
+ if pvc.Spec.VolumeMode == nil {
+ volumeMode := corev1.PersistentVolumeFilesystem
+ pvc.Spec.VolumeMode = &volumeMode
+ }
+ if pvc.Spec.Resources.Requests == nil {
+ pvc.Spec.Resources.Requests = corev1.ResourceList{}
+ }
+ // note: pvc.Spec.Resources.Requests.Storage() never returns nil
+ if pvc.Spec.Resources.Requests.Storage().IsZero() {
+ pvc.Spec.Resources.Requests[corev1.ResourceStorage] = repo.Spec.VolumeCapacity
}
if err := controllerutil.SetControllerReference(repo, pvc, r.Scheme); err != nil {
return fmt.Errorf("failed to set owner reference: %w", err)
@@ -501,6 +673,45 @@ func (r *BackupRepoReconciler) checkOrCreatePVC(
return err
}
+func constructToolConfigSecret(secret *corev1.Secret, content string) {
+ secret.Data = map[string][]byte{
+ "datasafed.conf": []byte(content),
+ }
+}
+
+func (r *BackupRepoReconciler) checkOrCreateToolConfigSecret(
+ reqCtx intctrlutil.RequestCtx, renderCtx renderContext,
+ repo *dpv1alpha1.BackupRepo, provider *storagev1alpha1.StorageProvider, namespace string) error {
+
+ secret := &corev1.Secret{}
+ secret.Name = repo.Status.ToolConfigSecretName
+ secret.Namespace = namespace
+ _, err := createObjectIfNotExist(reqCtx.Ctx, r.Client, secret,
+ func() error {
+ content, err := renderTemplate("tool-config", provider.Spec.DatasafedConfigTemplate, renderCtx)
+ if err != nil {
+ return fmt.Errorf("failed to render tool config template: %w", err)
+ }
+ constructToolConfigSecret(secret, content)
+
+ // add a referencing label
+ secret.Labels = map[string]string{
+ dataProtectionBackupRepoKey: repo.Name,
+ dataProtectionIsToolConfigKey: trueVal,
+ }
+ secret.Annotations = map[string]string{
+ dataProtectionTemplateValuesMD5AnnotationKey: renderCtx.Md5OfParameters(),
+ dataProtectionToolConfigTemplateMD5MD5AnnotationKey: md5Digest(provider.Spec.DatasafedConfigTemplate),
+ }
+ if err := controllerutil.SetControllerReference(repo, secret, r.Scheme); err != nil {
+ return fmt.Errorf("failed to set owner reference: %w", err)
+ }
+ return nil
+ })
+
+ return err
+}
+
func (r *BackupRepoReconciler) collectParameters(
reqCtx intctrlutil.RequestCtx, repo *dpv1alpha1.BackupRepo) (map[string]string, error) {
values := make(map[string]string)
@@ -562,7 +773,7 @@ func (r *BackupRepoReconciler) deleteExternalResources(
return err
}
- // delete derived secrets
+ // delete derived secrets (secret for CSI and tool configs)
if err := r.deleteSecrets(reqCtx, repo); err != nil {
return err
}
@@ -656,20 +867,20 @@ func (r *BackupRepoReconciler) deleteSecrets(reqCtx intctrlutil.RequestCtx, repo
return nil
}
-func (r *BackupRepoReconciler) mapBackupToRepo(obj client.Object) []ctrl.Request {
+func (r *BackupRepoReconciler) mapBackupToRepo(ctx context.Context, obj client.Object) []ctrl.Request {
backup := obj.(*dpv1alpha1.Backup)
repoName, ok := backup.Labels[dataProtectionBackupRepoKey]
if !ok {
return nil
}
// ignore failed backups
- if backup.Status.Phase == dpv1alpha1.BackupFailed {
+ if backup.Status.Phase == dpv1alpha1.BackupPhaseFailed {
return nil
}
// we should reconcile the BackupRepo when:
- // 1. the Backup needs a PVC which is not present and should be created by the BackupRepo.
+ // 1. the Backup needs to use the BackupRepo, but it's not ready for the namespace.
// 2. the Backup is being deleted, because it may block the deletion of the BackupRepo.
- shouldReconcileRepo := backup.Labels[dataProtectionNeedRepoPVCKey] == trueVal ||
+ shouldReconcileRepo := backup.Labels[dataProtectionWaitRepoPreparationKey] == trueVal ||
!backup.DeletionTimestamp.IsZero()
if shouldReconcileRepo {
return []ctrl.Request{{
@@ -679,11 +890,11 @@ func (r *BackupRepoReconciler) mapBackupToRepo(obj client.Object) []ctrl.Request
return nil
}
-func (r *BackupRepoReconciler) mapProviderToRepos(obj client.Object) []ctrl.Request {
+func (r *BackupRepoReconciler) mapProviderToRepos(ctx context.Context, obj client.Object) []ctrl.Request {
return r.providerRefMapper.mapToRequests(obj)
}
-func (r *BackupRepoReconciler) mapSecretToRepos(obj client.Object) []ctrl.Request {
+func (r *BackupRepoReconciler) mapSecretToRepos(ctx context.Context, obj client.Object) []ctrl.Request {
// check if the secret is created by this controller
owner := metav1.GetControllerOf(obj)
if owner != nil {
@@ -706,12 +917,9 @@ func (r *BackupRepoReconciler) mapSecretToRepos(obj client.Object) []ctrl.Reques
func (r *BackupRepoReconciler) SetupWithManager(mgr ctrl.Manager) error {
return ctrl.NewControllerManagedBy(mgr).
For(&dpv1alpha1.BackupRepo{}).
- Watches(&source.Kind{Type: &storagev1alpha1.StorageProvider{}},
- handler.EnqueueRequestsFromMapFunc(r.mapProviderToRepos)).
- Watches(&source.Kind{Type: &dpv1alpha1.Backup{}},
- handler.EnqueueRequestsFromMapFunc(r.mapBackupToRepo)).
- Watches(&source.Kind{Type: &corev1.Secret{}},
- handler.EnqueueRequestsFromMapFunc(r.mapSecretToRepos)).
+ Watches(&storagev1alpha1.StorageProvider{}, handler.EnqueueRequestsFromMapFunc(r.mapProviderToRepos)).
+ Watches(&dpv1alpha1.Backup{}, handler.EnqueueRequestsFromMapFunc(r.mapBackupToRepo)).
+ Watches(&corev1.Secret{}, handler.EnqueueRequestsFromMapFunc(r.mapSecretToRepos)).
Owns(&storagev1.StorageClass{}).
Owns(&corev1.PersistentVolumeClaim{}).
Complete(r)
@@ -721,9 +929,45 @@ func (r *BackupRepoReconciler) SetupWithManager(mgr ctrl.Manager) error {
// helper functions
// ============================================================================
+// dependencyError indicates that the error itself cannot be resolved
+// unless the dependent object is updated.
+type dependencyError struct {
+ msg string
+}
+
+func (e *dependencyError) Error() string {
+ return e.msg
+}
+
+func newDependencyError(msg string) error {
+ return &dependencyError{msg: msg}
+}
+
+func isDependencyError(err error) bool {
+ de, ok := err.(*dependencyError)
+ return ok || errors.As(err, &de)
+}
+
+func checkedRequeueWithError(err error, logger logr.Logger, msg string, keysAndValues ...interface{}) (reconcile.Result, error) {
+ if apierrors.IsNotFound(err) || isDependencyError(err) {
+ return intctrlutil.Reconciled()
+ }
+ return intctrlutil.RequeueWithError(err, logger, msg, keysAndValues...)
+}
+
type renderContext struct {
- Parameters map[string]string
- CSIDriverSecretRef corev1.SecretReference
+ Parameters map[string]string
+ CSIDriverSecretRef corev1.SecretReference
+ GeneratedStorageClassName string
+
+ md5OfParameters string
+}
+
+func (r *renderContext) Md5OfParameters() string {
+ if r.md5OfParameters == "" {
+ r.md5OfParameters = md5Digest(stableSerializeMap(r.Parameters))
+ }
+ return r.md5OfParameters
}
func renderTemplate(name, tpl string, rCtx renderContext) (string, error) {
@@ -737,19 +981,24 @@ func renderTemplate(name, tpl string, rCtx renderContext) (string, error) {
return b.String(), err
}
-func createObjectIfNotExist(
+func createOrUpdateObject[T any, PT generics.PObject[T]](
ctx context.Context,
c client.Client,
- obj client.Object,
- mutateFunc func() error) (created bool, err error) {
+ obj PT,
+ mutateFunc func() error,
+ shouldUpdate func() bool) (created bool, err error) {
key := client.ObjectKeyFromObject(obj)
err = c.Get(ctx, key, obj)
if err != nil && !apierrors.IsNotFound(err) {
- return false, fmt.Errorf("failed to check existence of object: %w", err)
+ return false, fmt.Errorf("failed to check existence of object %s: %w", key, err)
}
+ var patch client.Patch
if err == nil {
- // already exists
- return false, nil
+ // object already exists, check if it needs to be updated
+ if !shouldUpdate() {
+ return false, nil
+ }
+ patch = client.MergeFrom(PT(obj.DeepCopy()))
}
if mutateFunc != nil {
err := mutateFunc()
@@ -757,12 +1006,28 @@ func createObjectIfNotExist(
return false, err
}
}
- err = c.Create(ctx, obj)
- if err != nil {
- return false, fmt.Errorf("failed to create object %s: %w",
- client.ObjectKeyFromObject(obj), err)
+ if patch != nil {
+ err = c.Patch(ctx, obj, patch)
+ if err != nil {
+ err = fmt.Errorf("failed to patch object %s: %w", key, err)
+ }
+ return false, err
+ } else {
+ err = c.Create(ctx, obj)
+ if err != nil {
+ return false, fmt.Errorf("failed to create object %s: %w", key, err)
+ }
+ return true, nil
}
- return true, nil
+}
+
+func createObjectIfNotExist[T any, PT generics.PObject[T]](
+ ctx context.Context,
+ c client.Client,
+ obj PT,
+ mutateFunc func() error) (created bool, err error) {
+ noUpdate := func() bool { return false }
+ return createOrUpdateObject(ctx, c, obj, mutateFunc, noUpdate)
}
func setCondition(
@@ -794,6 +1059,25 @@ func updateCondition(
return c.Status().Patch(ctx, repo, patch)
}
+func updateAnnotations(ctx context.Context, c client.Client,
+ repo *dpv1alpha1.BackupRepo, annotations map[string]string) error {
+ patch := client.MergeFrom(repo.DeepCopy())
+ if repo.Annotations == nil {
+ repo.Annotations = make(map[string]string)
+ }
+ updated := false
+ for k, v := range annotations {
+ if curr, ok := repo.Annotations[k]; !ok || curr != v {
+ repo.Annotations[k] = v
+ updated = true
+ }
+ }
+ if !updated {
+ return nil
+ }
+ return c.Patch(ctx, repo, patch)
+}
+
func md5Digest(s string) string {
h := md5.New()
h.Write([]byte(s))
diff --git a/controllers/dataprotection/backuprepo_controller_test.go b/controllers/dataprotection/backuprepo_controller_test.go
index 804bf5743c0..fdee074a326 100644
--- a/controllers/dataprotection/backuprepo_controller_test.go
+++ b/controllers/dataprotection/backuprepo_controller_test.go
@@ -37,6 +37,7 @@ import (
dpv1alpha1 "github.com/apecloud/kubeblocks/apis/dataprotection/v1alpha1"
storagev1alpha1 "github.com/apecloud/kubeblocks/apis/storage/v1alpha1"
"github.com/apecloud/kubeblocks/internal/constant"
+ dptypes "github.com/apecloud/kubeblocks/internal/dataprotection/types"
intctrlutil "github.com/apecloud/kubeblocks/internal/generics"
testapps "github.com/apecloud/kubeblocks/internal/testutil/apps"
viper "github.com/apecloud/kubeblocks/internal/viperx"
@@ -163,7 +164,7 @@ var _ = Describe("BackupRepo controller", func() {
repoKey = client.ObjectKeyFromObject(repo)
}
- createStorageProviderSpec := func(mutateFunc func(repo *storagev1alpha1.StorageProvider)) {
+ createStorageProviderSpec := func(mutateFunc func(provider *storagev1alpha1.StorageProvider)) {
obj := &storagev1alpha1.StorageProvider{}
obj.GenerateName = "storageprovider-"
obj.Spec.CSIDriverName = defaultCSIDriverName
@@ -207,15 +208,41 @@ parameters:
obj.GenerateName = "backup-"
obj.Namespace = testCtx.DefaultNamespace
obj.Labels = map[string]string{
- dataProtectionBackupRepoKey: repoKey.Name,
- dataProtectionNeedRepoPVCKey: "true",
+ dataProtectionBackupRepoKey: repoKey.Name,
+ dataProtectionWaitRepoPreparationKey: trueVal,
}
- obj.Spec.BackupType = dpv1alpha1.BackupTypeSnapshot
+ obj.Spec.BackupMethod = "test-backup-method"
obj.Spec.BackupPolicyName = "default"
if mutateFunc != nil {
mutateFunc(obj)
}
- return testapps.CreateK8sResource(&testCtx, obj).(*dpv1alpha1.Backup)
+ backup := testapps.CreateK8sResource(&testCtx, obj).(*dpv1alpha1.Backup)
+ // updating the status of the Backup to COMPLETED, backup repo controller only
+ // handles for non-failed backups.
+ Eventually(func(g Gomega) {
+ obj := &dpv1alpha1.Backup{}
+ err := testCtx.Cli.Get(testCtx.Ctx, client.ObjectKeyFromObject(backup), obj)
+ g.Expect(err).ShouldNot(HaveOccurred())
+ if obj.Status.Phase == dpv1alpha1.BackupPhaseFailed {
+ // the controller will set the status to failed because
+ // essential objects (e.g. backup policy) are missed.
+ // we set the status to completed after that, to avoid conflict.
+ obj.Status.Phase = dpv1alpha1.BackupPhaseCompleted
+ err = testCtx.Cli.Status().Update(testCtx.Ctx, obj)
+ g.Expect(err).ShouldNot(HaveOccurred())
+ } else {
+ // check again
+ g.Expect(false).Should(BeTrue())
+ }
+ }).Should(Succeed())
+ return backup
+ }
+
+ getBackupRepo := func(g Gomega, key types.NamespacedName) *dpv1alpha1.BackupRepo {
+ repo := &dpv1alpha1.BackupRepo{}
+ err := testCtx.Cli.Get(testCtx.Ctx, key, repo)
+ g.Expect(err).ShouldNot(HaveOccurred())
+ return repo
}
deleteBackup := func(g Gomega, key types.NamespacedName) {
@@ -425,6 +452,10 @@ parameters:
By("checking the repo object again, it should be failed")
Eventually(testapps.CheckObj(&testCtx, repoKey, func(g Gomega, repo *dpv1alpha1.BackupRepo) {
g.Expect(repo.Status.Phase).Should(Equal(dpv1alpha1.BackupRepoFailed))
+ cond := meta.FindStatusCondition(repo.Status.Conditions, ConditionTypeParametersChecked)
+ g.Expect(cond).NotTo(BeNil())
+ g.Expect(cond.Status).Should(BeEquivalentTo(corev1.ConditionFalse))
+ g.Expect(cond.Reason).Should(Equal(ReasonCredentialSecretNotFound))
})).Should(Succeed())
})
@@ -439,7 +470,7 @@ parameters:
cond := meta.FindStatusCondition(repo.Status.Conditions, ConditionTypeStorageClassCreated)
g.Expect(cond).NotTo(BeNil())
g.Expect(cond.Status).Should(BeEquivalentTo(corev1.ConditionFalse))
- g.Expect(cond.Reason).Should(BeEquivalentTo(ReasonBadSecretTemplate))
+ g.Expect(cond.Reason).Should(BeEquivalentTo(ReasonPrepareCSISecretFailed))
g.Expect(cond.Message).Should(ContainSubstring(`function "bad" not defined`))
})).Should(Succeed())
})
@@ -455,7 +486,7 @@ parameters:
cond := meta.FindStatusCondition(repo.Status.Conditions, ConditionTypeStorageClassCreated)
g.Expect(cond).NotTo(BeNil())
g.Expect(cond.Status).Should(BeEquivalentTo(corev1.ConditionFalse))
- g.Expect(cond.Reason).Should(BeEquivalentTo(ReasonBadSecretTemplate))
+ g.Expect(cond.Reason).Should(BeEquivalentTo(ReasonPrepareCSISecretFailed))
g.Expect(cond.Message).Should(ContainSubstring(`cannot unmarshal string into Go value of type map[string]string`))
})).Should(Succeed())
})
@@ -473,7 +504,7 @@ parameters:
cond := meta.FindStatusCondition(repo.Status.Conditions, ConditionTypeStorageClassCreated)
g.Expect(cond).NotTo(BeNil())
g.Expect(cond.Status).Should(BeEquivalentTo(corev1.ConditionFalse))
- g.Expect(cond.Reason).Should(BeEquivalentTo(ReasonBadStorageClassTemplate))
+ g.Expect(cond.Reason).Should(BeEquivalentTo(ReasonPrepareStorageClassFailed))
g.Expect(cond.Message).Should(ContainSubstring(`function "bad" not defined`))
})).Should(Succeed())
})
@@ -491,7 +522,7 @@ parameters:
cond := meta.FindStatusCondition(repo.Status.Conditions, ConditionTypeStorageClassCreated)
g.Expect(cond).NotTo(BeNil())
g.Expect(cond.Status).Should(BeEquivalentTo(corev1.ConditionFalse))
- g.Expect(cond.Reason).Should(BeEquivalentTo(ReasonBadStorageClassTemplate))
+ g.Expect(cond.Reason).Should(BeEquivalentTo(ReasonPrepareStorageClassFailed))
g.Expect(cond.Message).Should(ContainSubstring(`cannot unmarshal string into Go value of type v1.StorageClass`))
})).Should(Succeed())
})
@@ -499,7 +530,7 @@ parameters:
createBackupAndCheckPVC := func(namespace string) (backup *dpv1alpha1.Backup, pvcName string) {
By("making sure the repo is ready")
Eventually(testapps.CheckObj(&testCtx, repoKey, func(g Gomega, repo *dpv1alpha1.BackupRepo) {
- g.Expect(repo.Status.Phase).Should(Equal(dpv1alpha1.BackupRepoReady))
+ g.Expect(repo.Status.Phase).Should(Equal(dpv1alpha1.BackupRepoReady), "%+v", repo)
g.Expect(repo.Status.BackupPVCName).ShouldNot(BeEmpty())
pvcName = repo.Status.BackupPVCName
})).Should(Succeed())
@@ -507,23 +538,6 @@ parameters:
backup = createBackupSpec(func(backup *dpv1alpha1.Backup) {
backup.Namespace = namespace
})
- By("updating the status of the Backup to completed")
- Eventually(func(g Gomega) {
- obj := &dpv1alpha1.Backup{}
- err := testCtx.Cli.Get(testCtx.Ctx, client.ObjectKeyFromObject(backup), obj)
- g.Expect(err).ShouldNot(HaveOccurred())
- if obj.Status.Phase == dpv1alpha1.BackupFailed {
- // the controller will set the status to failed because
- // essential objects (e.g. backup policy) are missed.
- // we set the status to completed after that, to avoid conflict.
- obj.Status.Phase = dpv1alpha1.BackupCompleted
- err = testCtx.Cli.Status().Update(testCtx.Ctx, obj)
- g.Expect(err).ShouldNot(HaveOccurred())
- } else {
- // check again
- g.Expect(false).Should(BeTrue())
- }
- }).Should(Succeed())
By("checking the PVC has been created in the namespace")
pvcKey := types.NamespacedName{
Name: pvcName,
@@ -543,6 +557,245 @@ parameters:
createBackupAndCheckPVC(namespace2)
})
+ Context("storage provider with PersistentVolumeClaimTemplate", func() {
+ It("should create a PVC in Backup's namespace (in default namespace)", func() {
+ By("setting the PersistentVolumeClaimTemplate")
+ createStorageProviderSpec(func(provider *storagev1alpha1.StorageProvider) {
+ provider.Spec.PersistentVolumeClaimTemplate = `
+apiVersion: v1
+kind: PersistentVolumeClaim
+metadata:
+ labels:
+ byPVCTemplate: "true"
+spec:
+ storageClassName: {{ .GeneratedStorageClassName }}
+ accessModes:
+ - ReadWriteOnce
+ resources:
+ volumeMode: Filesystem
+`
+ })
+ createBackupRepoSpec(nil)
+ _, pvcName := createBackupAndCheckPVC(testCtx.DefaultNamespace)
+
+ Eventually(testapps.CheckObj(&testCtx, types.NamespacedName{Name: pvcName, Namespace: testCtx.DefaultNamespace},
+ func(g Gomega, pvc *corev1.PersistentVolumeClaim) {
+ repo := getBackupRepo(g, repoKey)
+ g.Expect(pvc.Spec.StorageClassName).ShouldNot(BeNil())
+ g.Expect(*pvc.Spec.StorageClassName).Should(Equal(repo.Status.GeneratedStorageClassName))
+ g.Expect(pvc.Spec.Resources.Requests.Storage()).ShouldNot(BeNil())
+ g.Expect(pvc.Spec.Resources.Requests.Storage().String()).Should(Equal(repo.Spec.VolumeCapacity.String()))
+ g.Expect(pvc.Spec.AccessModes).Should(Equal([]corev1.PersistentVolumeAccessMode{corev1.ReadWriteOnce}))
+ g.Expect(pvc.Spec.VolumeMode).ShouldNot(BeNil())
+ g.Expect(*pvc.Spec.VolumeMode).Should(BeEquivalentTo(corev1.PersistentVolumeFilesystem))
+ g.Expect(pvc.Labels["byPVCTemplate"]).Should(Equal("true"))
+ })).Should(Succeed())
+ })
+
+ It("should fail if the PVC template is invalid", func() {
+ By("setting a invalid PersistentVolumeClaimTemplate")
+ Eventually(testapps.GetAndChangeObj(&testCtx, providerKey, func(provider *storagev1alpha1.StorageProvider) {
+ provider.Spec.PersistentVolumeClaimTemplate = `bad spec`
+ })).Should(Succeed())
+
+ By("checking repo's status")
+ Eventually(testapps.CheckObj(&testCtx, repoKey, func(g Gomega, repo *dpv1alpha1.BackupRepo) {
+ g.Expect(repo.Status.Phase, dpv1alpha1.BackupRepoFailed)
+ cond := meta.FindStatusCondition(repo.Status.Conditions, ConditionTypePVCTemplateChecked)
+ g.Expect(cond).NotTo(BeNil())
+ g.Expect(cond.Status).Should(BeEquivalentTo(corev1.ConditionFalse))
+ g.Expect(cond.Reason).Should(BeEquivalentTo(ReasonBadPVCTemplate))
+ })).Should(Succeed())
+ })
+ })
+
+ Context("storage provider contains only PersistentVolumeClaimTemplate", func() {
+ BeforeEach(func() {
+ createStorageProviderSpec(func(provider *storagev1alpha1.StorageProvider) {
+ provider.Spec.CSIDriverName = ""
+ provider.Spec.CSIDriverSecretTemplate = ""
+ provider.Spec.StorageClassTemplate = ""
+ provider.Spec.PersistentVolumeClaimTemplate = `
+spec:
+ storageClassName: some.storage.class
+ accessModes:
+ - ReadWriteOnce
+`
+ })
+ createBackupRepoSpec(nil)
+ })
+ It("should create the PVC based on the PersistentVolumeClaimTemplate", func() {
+ _, pvcName := createBackupAndCheckPVC(namespace2)
+ Eventually(testapps.CheckObj(&testCtx, types.NamespacedName{Name: pvcName, Namespace: namespace2},
+ func(g Gomega, pvc *corev1.PersistentVolumeClaim) {
+ g.Expect(pvc.Spec.StorageClassName).ShouldNot(BeNil())
+ g.Expect(*pvc.Spec.StorageClassName).Should(Equal("some.storage.class"))
+ g.Expect(pvc.Spec.AccessModes).Should(Equal([]corev1.PersistentVolumeAccessMode{corev1.ReadWriteOnce}))
+ g.Expect(pvc.Spec.VolumeMode).ShouldNot(BeNil())
+ g.Expect(*pvc.Spec.VolumeMode).Should(BeEquivalentTo(corev1.PersistentVolumeFilesystem))
+ g.Expect(pvc.Spec.Resources.Requests.Storage()).ShouldNot(BeNil())
+ g.Expect(pvc.Spec.Resources.Requests.Storage().String()).Should(Equal(repo.Spec.VolumeCapacity.String()))
+ })).Should(Succeed())
+ })
+ })
+
+ It("should fail if both StorageClassTemplate and PersistentVolumeClaimTemplate are empty", func() {
+ By("creating a storage provider with empty PersistentVolumeClaimTemplate and StorageClassTemplate")
+ createStorageProviderSpec(func(provider *storagev1alpha1.StorageProvider) {
+ provider.Spec.CSIDriverName = ""
+ provider.Spec.CSIDriverSecretTemplate = ""
+ provider.Spec.StorageClassTemplate = ""
+ provider.Spec.PersistentVolumeClaimTemplate = ""
+ })
+ By("creating a backup repo with the storage provider")
+ createBackupRepoSpec(nil)
+ By("checking repo's status")
+ Eventually(testapps.CheckObj(&testCtx, repoKey, func(g Gomega, repo *dpv1alpha1.BackupRepo) {
+ g.Expect(repo.Status.Phase).Should(BeEquivalentTo(dpv1alpha1.BackupRepoFailed))
+ cond := meta.FindStatusCondition(repo.Status.Conditions, ConditionTypeStorageProviderReady)
+ g.Expect(cond).NotTo(BeNil())
+ g.Expect(cond.Status).Should(BeEquivalentTo(corev1.ConditionFalse))
+ g.Expect(cond.Reason).Should(BeEquivalentTo(ReasonInvalidStorageProvider))
+ })).Should(Succeed())
+ })
+
+ Context("with AccessMethodTool", func() {
+ var repo *dpv1alpha1.BackupRepo
+ var backup *dpv1alpha1.Backup
+ var toolConfigSecretKey types.NamespacedName
+
+ BeforeEach(func() {
+ By("preparing")
+ createStorageProviderSpec(func(provider *storagev1alpha1.StorageProvider) {
+ provider.Spec.DatasafedConfigTemplate = `
+[storage]
+type=local
+key1={{ index .Parameters "key1" }}
+key2={{ index .Parameters "key2" }}
+cred-key1={{ index .Parameters "cred-key1" }}
+cred-key2={{ index .Parameters "cred-key2" }}
+`
+ })
+ createBackupRepoSpec(func(repo *dpv1alpha1.BackupRepo) {
+ repo.Spec.AccessMethod = dpv1alpha1.AccessMethodTool
+ })
+
+ Eventually(testapps.CheckObj(&testCtx, repoKey, func(g Gomega, obj *dpv1alpha1.BackupRepo) {
+ g.Expect(obj.Status.Phase).Should(Equal(dpv1alpha1.BackupRepoReady))
+ repo = obj
+ })).Should(Succeed())
+
+ backup = createBackupSpec(nil)
+ toolConfigSecretKey = types.NamespacedName{
+ Name: repo.Status.ToolConfigSecretName,
+ Namespace: backup.Namespace,
+ }
+ Eventually(testapps.CheckObjExists(&testCtx, toolConfigSecretKey, &corev1.Secret{}, true)).Should(Succeed())
+ })
+
+ It("should check that the storage provider has a non-empty datasafedConfigTemplate", func() {
+ By("preparing")
+ createStorageProviderSpec(func(provider *storagev1alpha1.StorageProvider) {
+ provider.Spec.DatasafedConfigTemplate = ""
+ })
+ createBackupRepoSpec(func(repo *dpv1alpha1.BackupRepo) {
+ repo.Spec.AccessMethod = dpv1alpha1.AccessMethodTool
+ })
+ By("checking")
+ Eventually(testapps.CheckObj(&testCtx, repoKey, func(g Gomega, repo *dpv1alpha1.BackupRepo) {
+ g.Expect(repo.Status.Phase).Should(Equal(dpv1alpha1.BackupRepoFailed))
+ cond := meta.FindStatusCondition(repo.Status.Conditions, ConditionTypeStorageProviderReady)
+ g.Expect(cond).NotTo(BeNil())
+ g.Expect(cond.Status).Should(BeEquivalentTo(corev1.ConditionFalse))
+ g.Expect(cond.Reason).Should(BeEquivalentTo(ReasonInvalidStorageProvider))
+ g.Expect(cond.Message).Should(ContainSubstring("DatasafedConfigTemplate is empty"))
+ })).Should(Succeed())
+ })
+
+ It("should fail if the datasafedConfigTemplate is invalid", func() {
+ By("preparing")
+ createStorageProviderSpec(func(provider *storagev1alpha1.StorageProvider) {
+ provider.Spec.DatasafedConfigTemplate = "bad template {{"
+ })
+ createBackupRepoSpec(func(repo *dpv1alpha1.BackupRepo) {
+ repo.Spec.AccessMethod = dpv1alpha1.AccessMethodTool
+ })
+ By("checking")
+ Eventually(testapps.CheckObj(&testCtx, repoKey, func(g Gomega, repo *dpv1alpha1.BackupRepo) {
+ g.Expect(repo.Status.Phase).Should(Equal(dpv1alpha1.BackupRepoFailed))
+ cond := meta.FindStatusCondition(repo.Status.Conditions, ConditionTypeToolConfigChecked)
+ g.Expect(cond).NotTo(BeNil())
+ g.Expect(cond.Status).Should(BeEquivalentTo(corev1.ConditionFalse))
+ g.Expect(cond.Reason).Should(BeEquivalentTo(ReasonBadToolConfigTemplate))
+ })).Should(Succeed())
+ })
+
+ It("should create the secret containing the tool config", func() {
+ Eventually(testapps.CheckObj(&testCtx, toolConfigSecretKey, func(g Gomega, secret *corev1.Secret) {
+ g.Expect(secret.Data).Should(HaveKeyWithValue("datasafed.conf", []byte(`
+[storage]
+type=local
+key1=val1
+key2=val2
+cred-key1=cred-val1
+cred-key2=cred-val2
+`)))
+ })).Should(Succeed())
+
+ By("creating a backup in namespace2")
+ createBackupSpec(func(backup *dpv1alpha1.Backup) {
+ backup.Namespace = namespace2
+ })
+ secretKey := types.NamespacedName{
+ Name: repo.Status.ToolConfigSecretName,
+ Namespace: namespace2,
+ }
+ Eventually(testapps.CheckObjExists(&testCtx, secretKey, &corev1.Secret{}, true)).Should(Succeed())
+ })
+
+ It("should update the content of the secret when the template or the value changes", func() {
+ By("changing the template")
+ Eventually(testapps.GetAndChangeObj(&testCtx, providerKey, func(provider *storagev1alpha1.StorageProvider) {
+ provider.Spec.DatasafedConfigTemplate += "new-item=new-value\n"
+ })).Should(Succeed())
+ Eventually(testapps.CheckObj(&testCtx, toolConfigSecretKey, func(g Gomega, secret *corev1.Secret) {
+ g.Expect(secret.Data).Should(HaveKeyWithValue("datasafed.conf", []byte(`
+[storage]
+type=local
+key1=val1
+key2=val2
+cred-key1=cred-val1
+cred-key2=cred-val2
+new-item=new-value
+`)))
+ })).Should(Succeed())
+
+ By("changing the value")
+ Eventually(testapps.GetAndChangeObj(&testCtx, repoKey, func(repo *dpv1alpha1.BackupRepo) {
+ repo.Spec.Config["key1"] = "changed-val1"
+ })).Should(Succeed())
+ Eventually(testapps.CheckObj(&testCtx, toolConfigSecretKey, func(g Gomega, secret *corev1.Secret) {
+ g.Expect(secret.Data).Should(HaveKeyWithValue("datasafed.conf", []byte(`
+[storage]
+type=local
+key1=changed-val1
+key2=val2
+cred-key1=cred-val1
+cred-key2=cred-val2
+new-item=new-value
+`)))
+ })).Should(Succeed())
+ })
+
+ It("should delete the secret when the repo is deleted", func() {
+ By("deleting the Backup and BackupRepo")
+ testapps.DeleteObject(&testCtx, client.ObjectKeyFromObject(backup), &dpv1alpha1.Backup{})
+ testapps.DeleteObject(&testCtx, repoKey, &dpv1alpha1.BackupRepo{})
+ By("checking the secret is deleted")
+ Eventually(testapps.CheckObjExists(&testCtx, toolConfigSecretKey, &corev1.Secret{}, false)).Should(Succeed())
+ })
+ })
+
It("should block the deletion of the BackupRepo if derived objects are not deleted", func() {
backup, pvcName := createBackupAndCheckPVC(namespace2)
@@ -628,7 +881,7 @@ parameters:
By("making the repo default")
Eventually(testapps.GetAndChangeObj(&testCtx, repoKey, func(repo *dpv1alpha1.BackupRepo) {
repo.Annotations = map[string]string{
- constant.DefaultBackupRepoAnnotationKey: trueVal,
+ dptypes.DefaultBackupRepoAnnotationKey: trueVal,
}
})).Should(Succeed())
By("checking the repo is default")
diff --git a/controllers/dataprotection/backupschedule_controller.go b/controllers/dataprotection/backupschedule_controller.go
new file mode 100644
index 00000000000..64f48815b9a
--- /dev/null
+++ b/controllers/dataprotection/backupschedule_controller.go
@@ -0,0 +1,244 @@
+/*
+Copyright (C) 2022-2023 ApeCloud Co., Ltd
+
+This file is part of KubeBlocks project
+
+This program is free software: you can redistribute it and/or modify
+it under the terms of the GNU Affero General Public License as published by
+the Free Software Foundation, either version 3 of the License, or
+(at your option) any later version.
+
+This program is distributed in the hope that it will be useful
+but WITHOUT ANY WARRANTY; without even the implied warranty of
+MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+GNU Affero General Public License for more details.
+
+You should have received a copy of the GNU Affero General Public License
+along with this program. If not, see .
+*/
+
+package dataprotection
+
+import (
+ "context"
+ "reflect"
+ "strings"
+ "time"
+
+ batchv1 "k8s.io/api/batch/v1"
+ corev1 "k8s.io/api/core/v1"
+ metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
+ k8sruntime "k8s.io/apimachinery/pkg/runtime"
+ "k8s.io/client-go/tools/record"
+ ctrl "sigs.k8s.io/controller-runtime"
+ "sigs.k8s.io/controller-runtime/pkg/client"
+ "sigs.k8s.io/controller-runtime/pkg/controller"
+ "sigs.k8s.io/controller-runtime/pkg/log"
+
+ dpv1alpha1 "github.com/apecloud/kubeblocks/apis/dataprotection/v1alpha1"
+ "github.com/apecloud/kubeblocks/internal/constant"
+ intctrlutil "github.com/apecloud/kubeblocks/internal/controllerutil"
+ dpbackup "github.com/apecloud/kubeblocks/internal/dataprotection/backup"
+ dptypes "github.com/apecloud/kubeblocks/internal/dataprotection/types"
+ dputils "github.com/apecloud/kubeblocks/internal/dataprotection/utils"
+ viper "github.com/apecloud/kubeblocks/internal/viperx"
+)
+
+// BackupScheduleReconciler reconciles a BackupSchedule object
+type BackupScheduleReconciler struct {
+ client.Client
+ Scheme *k8sruntime.Scheme
+ Recorder record.EventRecorder
+}
+
+// +kubebuilder:rbac:groups=dataprotection.kubeblocks.io,resources=backupschedules,verbs=get;list;watch;create;update;patch;delete
+// +kubebuilder:rbac:groups=dataprotection.kubeblocks.io,resources=backupschedules/status,verbs=get;update;patch
+// +kubebuilder:rbac:groups=dataprotection.kubeblocks.io,resources=backupschedules/finalizers,verbs=update
+
+// +kubebuilder:rbac:groups=batch,resources=cronjobs,verbs=get;list;watch;create;update;patch;delete
+// +kubebuilder:rbac:groups=batch,resources=cronjobs/status,verbs=get
+// +kubebuilder:rbac:groups=batch,resources=cronjobs/finalizers,verbs=update;patch
+
+// Reconcile is part of the main kubernetes reconciliation loop which aims to
+// move the current state of the backupschedule closer to the desired state.
+func (r *BackupScheduleReconciler) Reconcile(ctx context.Context, req ctrl.Request) (ctrl.Result, error) {
+ reqCtx := intctrlutil.RequestCtx{
+ Ctx: ctx,
+ Req: req,
+ Log: log.FromContext(ctx).WithValues("backupSchedule", req.NamespacedName),
+ Recorder: r.Recorder,
+ }
+
+ backupSchedule := &dpv1alpha1.BackupSchedule{}
+ if err := r.Client.Get(reqCtx.Ctx, reqCtx.Req.NamespacedName, backupSchedule); err != nil {
+ return intctrlutil.CheckedRequeueWithError(err, reqCtx.Log, "")
+ }
+
+ original := backupSchedule.DeepCopy()
+
+ // handle finalizer
+ res, err := intctrlutil.HandleCRDeletion(reqCtx, r, backupSchedule, dptypes.DataProtectionFinalizerName, func() (*ctrl.Result, error) {
+ return nil, r.deleteExternalResources(reqCtx, backupSchedule)
+ })
+ if res != nil {
+ return *res, err
+ }
+
+ // try to remove expired or oldest backups, triggered by cronjob controller
+ // TODO(ldm): another garbage collection controller to remove expired backups
+ if err = r.removeExpiredBackups(reqCtx); err != nil {
+ return r.patchStatusFailed(reqCtx, backupSchedule, "RemoveExpiredBackupsFailed", err)
+ }
+
+ if err = r.handleSchedule(reqCtx, backupSchedule); err != nil {
+ return r.patchStatusFailed(reqCtx, backupSchedule, "HandleBackupScheduleFailed", err)
+ }
+
+ return r.patchStatusAvailable(reqCtx, original, backupSchedule)
+}
+
+// SetupWithManager sets up the controller with the Manager.
+func (r *BackupScheduleReconciler) SetupWithManager(mgr ctrl.Manager) error {
+ return ctrl.NewControllerManagedBy(mgr).
+ For(&dpv1alpha1.BackupSchedule{}).
+ Owns(&batchv1.CronJob{}).
+ WithOptions(controller.Options{
+ MaxConcurrentReconciles: viper.GetInt(maxConcurDataProtectionReconKey),
+ }).
+ Complete(r)
+}
+
+func (r *BackupScheduleReconciler) deleteExternalResources(
+ reqCtx intctrlutil.RequestCtx,
+ backupSchedule *dpv1alpha1.BackupSchedule) error {
+ // delete cronjob resource
+ cronJobList := &batchv1.CronJobList{}
+ if err := r.Client.List(reqCtx.Ctx, cronJobList,
+ client.InNamespace(backupSchedule.Namespace),
+ client.MatchingLabels{
+ dataProtectionLabelBackupScheduleKey: backupSchedule.Name,
+ constant.AppManagedByLabelKey: constant.AppName,
+ },
+ ); err != nil {
+ return err
+ }
+ for _, cronjob := range cronJobList.Items {
+ if err := dputils.RemoveDataProtectionFinalizer(reqCtx.Ctx, r.Client, &cronjob); err != nil {
+ return err
+ }
+ if err := intctrlutil.BackgroundDeleteObject(r.Client, reqCtx.Ctx, &cronjob); err != nil {
+ // failed delete k8s job, return error info.
+ return err
+ }
+ }
+ // notice running backup to completed
+ // TODO(ldm): is it necessary to notice running backup to completed?
+ backup := &dpv1alpha1.Backup{}
+ for _, s := range backupSchedule.Spec.Schedules {
+ backupKey := client.ObjectKey{
+ Namespace: backupSchedule.Namespace,
+ Name: dpbackup.GenerateCRNameByBackupSchedule(backupSchedule, s.BackupMethod),
+ }
+ if err := r.Client.Get(reqCtx.Ctx, backupKey, backup); err != nil {
+ if client.IgnoreNotFound(err) == nil {
+ continue
+ }
+ return err
+ }
+ patch := client.MergeFrom(backup.DeepCopy())
+ backup.Status.Phase = dpv1alpha1.BackupPhaseCompleted
+ backup.Status.CompletionTimestamp = &metav1.Time{Time: time.Now().UTC()}
+ if err := r.Client.Status().Patch(reqCtx.Ctx, backup, patch); err != nil {
+ return err
+ }
+ }
+ return nil
+}
+
+// patchStatusAvailable patches backup policy status phase to available.
+func (r *BackupScheduleReconciler) patchStatusAvailable(reqCtx intctrlutil.RequestCtx,
+ origin, backupSchedule *dpv1alpha1.BackupSchedule) (ctrl.Result, error) {
+ if !reflect.DeepEqual(origin.Spec, backupSchedule.Spec) {
+ if err := r.Client.Update(reqCtx.Ctx, backupSchedule); err != nil {
+ return intctrlutil.CheckedRequeueWithError(err, reqCtx.Log, "")
+ }
+ }
+ // update status phase
+ if backupSchedule.Status.Phase != dpv1alpha1.BackupSchedulePhaseAvailable ||
+ backupSchedule.Status.ObservedGeneration != backupSchedule.Generation {
+ patch := client.MergeFrom(backupSchedule.DeepCopy())
+ backupSchedule.Status.ObservedGeneration = backupSchedule.Generation
+ backupSchedule.Status.Phase = dpv1alpha1.BackupSchedulePhaseAvailable
+ backupSchedule.Status.FailureReason = ""
+ if err := r.Client.Status().Patch(reqCtx.Ctx, backupSchedule, patch); err != nil {
+ return intctrlutil.CheckedRequeueWithError(err, reqCtx.Log, "")
+ }
+ }
+ return intctrlutil.Reconciled()
+}
+
+// patchStatusFailed patches backup policy status phase to failed.
+func (r *BackupScheduleReconciler) patchStatusFailed(reqCtx intctrlutil.RequestCtx,
+ backupSchedule *dpv1alpha1.BackupSchedule,
+ reason string,
+ err error) (ctrl.Result, error) {
+ if intctrlutil.IsTargetError(err, intctrlutil.ErrorTypeRequeue) {
+ return intctrlutil.RequeueAfter(reconcileInterval, reqCtx.Log, "")
+ }
+ backupScheduleDeepCopy := backupSchedule.DeepCopy()
+ backupSchedule.Status.Phase = dpv1alpha1.BackupSchedulePhaseFailed
+ backupSchedule.Status.FailureReason = err.Error()
+ if !reflect.DeepEqual(backupSchedule.Status, backupScheduleDeepCopy.Status) {
+ if patchErr := r.Client.Status().Patch(reqCtx.Ctx, backupSchedule, client.MergeFrom(backupScheduleDeepCopy)); patchErr != nil {
+ return intctrlutil.RequeueWithError(patchErr, reqCtx.Log, "")
+ }
+ }
+ r.Recorder.Event(backupSchedule, corev1.EventTypeWarning, reason, err.Error())
+ return intctrlutil.RequeueWithError(err, reqCtx.Log, "")
+}
+
+func (r *BackupScheduleReconciler) removeExpiredBackups(reqCtx intctrlutil.RequestCtx) error {
+ backups := dpv1alpha1.BackupList{}
+ if err := r.Client.List(reqCtx.Ctx, &backups,
+ client.InNamespace(reqCtx.Req.Namespace)); err != nil {
+ return err
+ }
+
+ now := metav1.Now()
+ for _, item := range backups.Items {
+ // ignore retained backup.
+ if strings.EqualFold(item.GetLabels()[constant.BackupProtectionLabelKey], constant.BackupRetain) {
+ continue
+ }
+
+ // ignore backup which is not expired.
+ if item.Status.Expiration == nil || !item.Status.Expiration.Before(&now) {
+ continue
+ }
+
+ // delete expired backup.
+ if err := intctrlutil.BackgroundDeleteObject(r.Client, reqCtx.Ctx, &item); err != nil {
+ // failed delete backups, return error info.
+ return err
+ }
+ }
+ return nil
+}
+
+// handleSchedule handles backup schedules for different backup method.
+func (r *BackupScheduleReconciler) handleSchedule(
+ reqCtx intctrlutil.RequestCtx,
+ backupSchedule *dpv1alpha1.BackupSchedule) error {
+ backupPolicy, err := getBackupPolicyByName(reqCtx, r.Client, backupSchedule.Spec.BackupPolicyName)
+ if err != nil {
+ return err
+ }
+ scheduler := dpbackup.Scheduler{
+ RequestCtx: reqCtx,
+ BackupSchedule: backupSchedule,
+ BackupPolicy: backupPolicy,
+ Client: r.Client,
+ Scheme: r.Scheme,
+ }
+ return scheduler.Schedule()
+}
diff --git a/controllers/dataprotection/backupschedule_controller_test.go b/controllers/dataprotection/backupschedule_controller_test.go
new file mode 100644
index 00000000000..c96f6cae4ba
--- /dev/null
+++ b/controllers/dataprotection/backupschedule_controller_test.go
@@ -0,0 +1,282 @@
+/*
+Copyright (C) 2022-2023 ApeCloud Co., Ltd
+
+This file is part of KubeBlocks project
+
+This program is free software: you can redistribute it and/or modify
+it under the terms of the GNU Affero General Public License as published by
+the Free Software Foundation, either version 3 of the License, or
+(at your option) any later version.
+
+This program is distributed in the hope that it will be useful
+but WITHOUT ANY WARRANTY; without even the implied warranty of
+MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+GNU Affero General Public License for more details.
+
+You should have received a copy of the GNU Affero General Public License
+along with this program. If not, see .
+*/
+
+package dataprotection
+
+import (
+ "fmt"
+ "time"
+
+ . "github.com/onsi/ginkgo/v2"
+ . "github.com/onsi/gomega"
+ batchv1 "k8s.io/api/batch/v1"
+ metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
+ "sigs.k8s.io/controller-runtime/pkg/client"
+
+ dpv1alpha1 "github.com/apecloud/kubeblocks/apis/dataprotection/v1alpha1"
+ dpbackup "github.com/apecloud/kubeblocks/internal/dataprotection/backup"
+ "github.com/apecloud/kubeblocks/internal/dataprotection/utils/boolptr"
+ "github.com/apecloud/kubeblocks/internal/generics"
+ testapps "github.com/apecloud/kubeblocks/internal/testutil/apps"
+ testdp "github.com/apecloud/kubeblocks/internal/testutil/dataprotection"
+)
+
+var _ = Describe("Backup Schedule Controller", func() {
+ cleanEnv := func() {
+ // must wait till resources deleted and no longer existed before the testcases start,
+ // otherwise if later it needs to create some new resource objects with the same name,
+ // in race conditions, it will find the existence of old objects, resulting failure to
+ // create the new objects.
+ By("clean resources")
+ // delete rest mocked objects
+ inNS := client.InNamespace(testCtx.DefaultNamespace)
+ ml := client.HasLabels{testCtx.TestObjLabelKey}
+
+ // namespaced
+ testapps.ClearResources(&testCtx, generics.ClusterSignature, inNS, ml)
+ testapps.ClearResources(&testCtx, generics.PodSignature, inNS, ml)
+ testapps.ClearResources(&testCtx, generics.SecretSignature, inNS, ml)
+ testapps.ClearResourcesWithRemoveFinalizerOption(&testCtx, generics.BackupPolicySignature, true, inNS)
+ testapps.ClearResourcesWithRemoveFinalizerOption(&testCtx, generics.BackupScheduleSignature, true, inNS)
+ testapps.ClearResourcesWithRemoveFinalizerOption(&testCtx, generics.BackupSignature, true, inNS)
+
+ // wait all backup to be deleted, otherwise the controller maybe create
+ // job to delete the backup between the ClearResources function delete
+ // the job and get the job list, resulting the ClearResources panic.
+ Eventually(testapps.List(&testCtx, generics.BackupSignature, inNS)).Should(HaveLen(0))
+
+ testapps.ClearResourcesWithRemoveFinalizerOption(&testCtx, generics.JobSignature, true, inNS)
+ testapps.ClearResourcesWithRemoveFinalizerOption(&testCtx, generics.PersistentVolumeClaimSignature, true, inNS)
+ testapps.ClearResourcesWithRemoveFinalizerOption(&testCtx, generics.CronJobSignature, true, inNS)
+
+ // non-namespaced
+ testapps.ClearResourcesWithRemoveFinalizerOption(&testCtx, generics.ActionSetSignature, true, ml)
+ testapps.ClearResources(&testCtx, generics.StorageClassSignature, ml)
+ testapps.ClearResourcesWithRemoveFinalizerOption(&testCtx, generics.BackupRepoSignature, true, ml)
+ testapps.ClearResources(&testCtx, generics.StorageProviderSignature, ml)
+ }
+
+ BeforeEach(func() {
+ cleanEnv()
+ _ = testdp.NewFakeCluster(&testCtx)
+ })
+
+ AfterEach(cleanEnv)
+
+ When("creating backup schedule with default settings", func() {
+ var (
+ backupPolicy *dpv1alpha1.BackupPolicy
+ )
+
+ getCronjobKey := func(backupSchedule *dpv1alpha1.BackupSchedule,
+ method string) client.ObjectKey {
+ return client.ObjectKey{
+ Name: dpbackup.GenerateCRNameByBackupSchedule(backupSchedule, method),
+ Namespace: backupPolicy.Namespace,
+ }
+ }
+
+ getJobKey := func(backup *dpv1alpha1.Backup) client.ObjectKey {
+ return client.ObjectKey{
+ Name: dpbackup.GenerateBackupJobName(backup, dpbackup.BackupDataJobNamePrefix),
+ Namespace: backup.Namespace,
+ }
+ }
+
+ BeforeEach(func() {
+ By("creating an actionSet")
+ actionSet := testdp.NewFakeActionSet(&testCtx)
+
+ By("creating storage provider")
+ _ = testdp.NewFakeStorageProvider(&testCtx, nil)
+
+ By("creating backup repo")
+ _, _ = testdp.NewFakeBackupRepo(&testCtx, nil)
+
+ By("By creating a backupPolicy from actionSet " + actionSet.Name)
+ backupPolicy = testdp.NewFakeBackupPolicy(&testCtx, nil)
+ })
+
+ AfterEach(func() {
+ })
+
+ Context("creates a backup schedule", func() {
+ var (
+ backupNamePrefix = "schedule-test-backup-"
+ backupSchedule *dpv1alpha1.BackupSchedule
+ backupScheduleKey client.ObjectKey
+ )
+ BeforeEach(func() {
+ By("creating a backupSchedule")
+ backupSchedule = testdp.NewFakeBackupSchedule(&testCtx, nil)
+ backupScheduleKey = client.ObjectKeyFromObject(backupSchedule)
+ })
+
+ It("should success", func() {
+ By("checking backupSchedule status, should be available")
+ Eventually(testapps.CheckObj(&testCtx, backupScheduleKey, func(g Gomega, fetched *dpv1alpha1.BackupSchedule) {
+ g.Expect(fetched.Status.Phase).To(Equal(dpv1alpha1.BackupSchedulePhaseAvailable))
+ })).Should(Succeed())
+
+ By("checking cronjob, should not exist because all schedule policies of methods are disabled")
+ Eventually(testapps.CheckObjExists(&testCtx, getCronjobKey(backupSchedule, testdp.BackupMethodName),
+ &batchv1.CronJob{}, false)).Should(Succeed())
+ Eventually(testapps.CheckObjExists(&testCtx, getCronjobKey(backupSchedule, testdp.VSBackupMethodName),
+ &batchv1.CronJob{}, false)).Should(Succeed())
+
+ By(fmt.Sprintf("enabling %s method schedule", testdp.BackupMethodName))
+ testdp.EnableBackupSchedule(&testCtx, backupSchedule, testdp.BackupMethodName)
+
+ By("checking cronjob, should exist one cronjob to create backup")
+ Eventually(testapps.CheckObj(&testCtx, getCronjobKey(backupSchedule, testdp.BackupMethodName), func(g Gomega, fetched *batchv1.CronJob) {
+ schedulePolicy := dpbackup.GetSchedulePolicyByMethod(backupSchedule, testdp.BackupMethodName)
+ g.Expect(boolptr.IsSetToTrue(schedulePolicy.Enabled)).To(BeTrue())
+ g.Expect(fetched.Spec.Schedule).To(Equal(schedulePolicy.CronExpression))
+ g.Expect(fetched.Spec.StartingDeadlineSeconds).ShouldNot(BeNil())
+ g.Expect(*fetched.Spec.StartingDeadlineSeconds).To(Equal(getStartingDeadlineSeconds(backupSchedule)))
+ })).Should(Succeed())
+ })
+
+ It("delete expired backups", func() {
+ now := metav1.Now()
+ backupStatus := dpv1alpha1.BackupStatus{
+ Phase: dpv1alpha1.BackupPhaseCompleted,
+ Expiration: &now,
+ StartTimestamp: &now,
+ CompletionTimestamp: &now,
+ }
+
+ autoBackupLabel := map[string]string{
+ dataProtectionLabelAutoBackupKey: "true",
+ dataProtectionLabelBackupPolicyKey: testdp.BackupPolicyName,
+ dataProtectionLabelBackupMethodKey: testdp.BackupMethodName,
+ }
+
+ createBackup := func(name string) *dpv1alpha1.Backup {
+ return testdp.NewBackupFactory(testCtx.DefaultNamespace, name).
+ WithRandomName().AddLabelsInMap(autoBackupLabel).
+ SetBackupPolicyName(testdp.BackupPolicyName).
+ SetBackupMethod(testdp.BackupMethodName).
+ Create(&testCtx).GetObject()
+ }
+
+ checkBackupCompleted := func(key client.ObjectKey) {
+ Eventually(testapps.CheckObj(&testCtx, key,
+ func(g Gomega, fetched *dpv1alpha1.Backup) {
+ g.Expect(fetched.Status.Phase).To(Equal(dpv1alpha1.BackupPhaseCompleted))
+ })).Should(Succeed())
+ }
+
+ By("create an expired backup")
+ backupExpired := createBackup(backupNamePrefix + "expired")
+
+ By("create 1st backup")
+ backupOutLimit1 := createBackup(backupNamePrefix + "1")
+
+ By("create 2nd backup")
+ backupOutLimit2 := createBackup(backupNamePrefix + "2")
+
+ By("waiting expired backup completed")
+ expiredKey := client.ObjectKeyFromObject(backupExpired)
+ testdp.PatchK8sJobStatus(&testCtx, getJobKey(backupExpired), batchv1.JobComplete)
+ checkBackupCompleted(expiredKey)
+
+ By("mock update expired backup status to expire")
+ backupStatus.Expiration = &metav1.Time{Time: now.Add(-time.Hour * 24)}
+ backupStatus.StartTimestamp = backupStatus.Expiration
+ testdp.PatchBackupStatus(&testCtx, client.ObjectKeyFromObject(backupExpired), backupStatus)
+
+ By("waiting 1st backup completed")
+ outLimit1Key := client.ObjectKeyFromObject(backupOutLimit1)
+ testdp.PatchK8sJobStatus(&testCtx, getJobKey(backupOutLimit1), batchv1.JobComplete)
+ checkBackupCompleted(outLimit1Key)
+
+ By("mock 1st backup not to expire")
+ backupStatus.Expiration = &metav1.Time{Time: now.Add(time.Hour * 24)}
+ backupStatus.StartTimestamp = &metav1.Time{Time: now.Add(time.Hour)}
+ testdp.PatchBackupStatus(&testCtx, client.ObjectKeyFromObject(backupOutLimit1), backupStatus)
+
+ By("waiting 2nd backup completed")
+ outLimit2Key := client.ObjectKeyFromObject(backupOutLimit2)
+ testdp.PatchK8sJobStatus(&testCtx, getJobKey(backupOutLimit2), batchv1.JobComplete)
+ checkBackupCompleted(outLimit2Key)
+
+ By("mock 2nd backup not to expire")
+ backupStatus.Expiration = &metav1.Time{Time: now.Add(time.Hour * 24)}
+ backupStatus.StartTimestamp = &metav1.Time{Time: now.Add(time.Hour * 2)}
+ testdp.PatchBackupStatus(&testCtx, client.ObjectKeyFromObject(backupOutLimit2), backupStatus)
+
+ By("patch backup schedule to trigger the controller to delete expired backup")
+ Eventually(testapps.GetAndChangeObj(&testCtx, backupScheduleKey, func(fetched *dpv1alpha1.BackupSchedule) {
+ fetched.Spec.Schedules[0].RetentionPeriod = "1d"
+ })).Should(Succeed())
+
+ By("retain the latest backup")
+ Eventually(testapps.List(&testCtx, generics.BackupSignature,
+ client.MatchingLabels(autoBackupLabel),
+ client.InNamespace(backupPolicy.Namespace))).Should(HaveLen(2))
+ })
+ })
+
+ Context("creates a backup schedule with empty schedule", func() {
+ It("should fail when create a backupSchedule without nil schedule policy", func() {
+ backupScheduleObj := testdp.NewBackupScheduleFactory(testCtx.DefaultNamespace, testdp.BackupScheduleName).
+ SetBackupPolicyName(testdp.BackupPolicyName).
+ SetSchedules(nil).
+ GetObject()
+ Expect(testCtx.CheckedCreateObj(testCtx.Ctx, backupScheduleObj)).Should(HaveOccurred())
+ })
+
+ It("should fail when create a backupSchedule without empty schedule policy", func() {
+ backupScheduleObj := testdp.NewBackupScheduleFactory(testCtx.DefaultNamespace, testdp.BackupScheduleName).
+ SetBackupPolicyName(testdp.BackupPolicyName).
+ GetObject()
+ Expect(testCtx.CheckedCreateObj(testCtx.Ctx, backupScheduleObj)).Should(HaveOccurred())
+ })
+ })
+
+ Context("creates a backup schedule with invalid field", func() {
+ var (
+ backupScheduleKey client.ObjectKey
+ backupSchedule *dpv1alpha1.BackupSchedule
+ )
+
+ BeforeEach(func() {
+ By("creating a backupSchedule")
+ backupSchedule = testdp.NewFakeBackupSchedule(&testCtx, func(schedule *dpv1alpha1.BackupSchedule) {
+ schedule.Spec.Schedules[0].CronExpression = "invalid"
+ })
+ backupScheduleKey = client.ObjectKeyFromObject(backupSchedule)
+ })
+
+ It("should fail", func() {
+ Eventually(testapps.CheckObj(&testCtx, backupScheduleKey, func(g Gomega, fetched *dpv1alpha1.BackupSchedule) {
+ g.Expect(fetched.Status.Phase).NotTo(Equal(dpv1alpha1.BackupSchedulePhaseAvailable))
+ })).Should(Succeed())
+ })
+ })
+ })
+})
+
+func getStartingDeadlineSeconds(backupSchedule *dpv1alpha1.BackupSchedule) int64 {
+ if backupSchedule.Spec.StartingDeadlineMinutes == nil {
+ return 0
+ }
+ return *backupSchedule.Spec.StartingDeadlineMinutes * 60
+}
diff --git a/controllers/dataprotection/backuptool_controller.go b/controllers/dataprotection/backuptool_controller.go
deleted file mode 100644
index 768b315bebe..00000000000
--- a/controllers/dataprotection/backuptool_controller.go
+++ /dev/null
@@ -1,103 +0,0 @@
-/*
-Copyright (C) 2022-2023 ApeCloud Co., Ltd
-
-This file is part of KubeBlocks project
-
-This program is free software: you can redistribute it and/or modify
-it under the terms of the GNU Affero General Public License as published by
-the Free Software Foundation, either version 3 of the License, or
-(at your option) any later version.
-
-This program is distributed in the hope that it will be useful
-but WITHOUT ANY WARRANTY; without even the implied warranty of
-MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
-GNU Affero General Public License for more details.
-
-You should have received a copy of the GNU Affero General Public License
-along with this program. If not, see .
-*/
-
-package dataprotection
-
-import (
- "context"
-
- "k8s.io/apimachinery/pkg/runtime"
- "k8s.io/client-go/tools/record"
- ctrl "sigs.k8s.io/controller-runtime"
- "sigs.k8s.io/controller-runtime/pkg/client"
- "sigs.k8s.io/controller-runtime/pkg/controller"
- "sigs.k8s.io/controller-runtime/pkg/log"
-
- dataprotectionv1alpha1 "github.com/apecloud/kubeblocks/apis/dataprotection/v1alpha1"
- intctrlutil "github.com/apecloud/kubeblocks/internal/controllerutil"
- viper "github.com/apecloud/kubeblocks/internal/viperx"
-)
-
-// BackupToolReconciler reconciles a BackupTool object
-type BackupToolReconciler struct {
- client.Client
- Scheme *runtime.Scheme
- Recorder record.EventRecorder
-}
-
-// +kubebuilder:rbac:groups=dataprotection.kubeblocks.io,resources=backuptools,verbs=get;list;watch;create;update;patch;delete
-// +kubebuilder:rbac:groups=dataprotection.kubeblocks.io,resources=backuptools/status,verbs=get;update;patch
-// +kubebuilder:rbac:groups=dataprotection.kubeblocks.io,resources=backuptools/finalizers,verbs=update
-
-// Reconcile is part of the main kubernetes reconciliation loop which aims to
-// move the current state of the cluster closer to the desired state.
-// TODO(user): Modify the Reconcile function to compare the state specified by
-// the BackupTool object against the actual cluster state, and then
-// perform operations to make the cluster state reflect the state specified by
-// the user.
-//
-// For more details, check Reconcile and its Result here:
-// - https://pkg.go.dev/sigs.k8s.io/controller-runtime@v0.12.1/pkg/reconcile
-func (r *BackupToolReconciler) Reconcile(ctx context.Context, req ctrl.Request) (ctrl.Result, error) {
- _ = log.FromContext(ctx)
- // NOTES:
- // setup common request context
- reqCtx := intctrlutil.RequestCtx{
- Ctx: ctx,
- Req: req,
- Log: log.FromContext(ctx).WithValues("backupTool", req.NamespacedName),
- Recorder: r.Recorder,
- }
-
- backupTool := &dataprotectionv1alpha1.BackupTool{}
- if err := r.Client.Get(reqCtx.Ctx, reqCtx.Req.NamespacedName, backupTool); err != nil {
- return intctrlutil.CheckedRequeueWithError(err, reqCtx.Log, "")
- }
-
- // handle finalizer
- res, err := intctrlutil.HandleCRDeletion(reqCtx, r, backupTool, dataProtectionFinalizerName, func() (*ctrl.Result, error) {
- return nil, r.deleteExternalResources(reqCtx, backupTool)
- })
- if res != nil {
- return *res, err
- }
- // TODO(user): your logic here
-
- return ctrl.Result{}, nil
-}
-
-// SetupWithManager sets up the controller with the Manager.
-func (r *BackupToolReconciler) SetupWithManager(mgr ctrl.Manager) error {
- return ctrl.NewControllerManagedBy(mgr).
- For(&dataprotectionv1alpha1.BackupTool{}).
- WithOptions(controller.Options{
- MaxConcurrentReconciles: viper.GetInt(maxConcurDataProtectionReconKey),
- }).
- Complete(r)
-}
-
-func (r *BackupToolReconciler) deleteExternalResources(reqCtx intctrlutil.RequestCtx, backupTool *dataprotectionv1alpha1.BackupTool) error {
- //
- // delete any external resources associated with the cronJob
- //
- // Ensure that delete implementation is idempotent and safe to invoke
- // multiple times for same object.
-
- return nil
-}
diff --git a/controllers/dataprotection/cronjob_controller.go b/controllers/dataprotection/cronjob_controller.go
deleted file mode 100644
index 7bad6b9f2ab..00000000000
--- a/controllers/dataprotection/cronjob_controller.go
+++ /dev/null
@@ -1,96 +0,0 @@
-/*
-Copyright (C) 2022-2023 ApeCloud Co., Ltd
-
-This file is part of KubeBlocks project
-
-This program is free software: you can redistribute it and/or modify
-it under the terms of the GNU Affero General Public License as published by
-the Free Software Foundation, either version 3 of the License, or
-(at your option) any later version.
-
-This program is distributed in the hope that it will be useful
-but WITHOUT ANY WARRANTY; without even the implied warranty of
-MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
-GNU Affero General Public License for more details.
-
-You should have received a copy of the GNU Affero General Public License
-along with this program. If not, see .
-*/
-
-package dataprotection
-
-import (
- "context"
-
- batchv1 "k8s.io/api/batch/v1"
- "k8s.io/apimachinery/pkg/runtime"
- "k8s.io/apimachinery/pkg/types"
- "k8s.io/client-go/tools/record"
- ctrl "sigs.k8s.io/controller-runtime"
- "sigs.k8s.io/controller-runtime/pkg/client"
- "sigs.k8s.io/controller-runtime/pkg/log"
- "sigs.k8s.io/controller-runtime/pkg/predicate"
-
- dataprotectionv1alpha1 "github.com/apecloud/kubeblocks/apis/dataprotection/v1alpha1"
- intctrlutil "github.com/apecloud/kubeblocks/internal/controllerutil"
-)
-
-// CronJobReconciler reconciles a cronjob object
-type CronJobReconciler struct {
- client.Client
- Scheme *runtime.Scheme
- Recorder record.EventRecorder
-}
-
-// +kubebuilder:rbac:groups=batch,resources=cronjobs,verbs=get;list;watch;create;update;patch;delete
-// +kubebuilder:rbac:groups=batch,resources=cronjobs/status,verbs=get
-// +kubebuilder:rbac:groups=batch,resources=cronjobs/finalizers,verbs=update
-
-// Reconcile is part of the main kubernetes reconciliation loop which aims to
-// move the current state of the cluster closer to the desired state.
-//
-// For more details, check Reconcile and its Result here:
-// - https://pkg.go.dev/sigs.k8s.io/controller-runtime@v0.14.4/pkg/reconcile
-func (r *CronJobReconciler) Reconcile(ctx context.Context, req ctrl.Request) (ctrl.Result, error) {
- var (
- cronJob = &batchv1.CronJob{}
- backupPolicy = &dataprotectionv1alpha1.BackupPolicy{}
- err error
- )
-
- reqCtx := intctrlutil.RequestCtx{
- Ctx: ctx,
- Req: req,
- Log: log.FromContext(ctx).WithValues("cronJob", req.NamespacedName),
- }
-
- if err = r.Client.Get(reqCtx.Ctx, reqCtx.Req.NamespacedName, cronJob); err != nil {
- return intctrlutil.CheckedRequeueWithError(err, reqCtx.Log, "")
- }
-
- backupPolicyKey := types.NamespacedName{
- Namespace: cronJob.Annotations["kubeblocks.io/backup-namespace"],
- Name: cronJob.Labels[dataProtectionLabelBackupPolicyKey],
- }
- if err = r.Client.Get(reqCtx.Ctx, backupPolicyKey, backupPolicy); err != nil {
- return intctrlutil.CheckedRequeueWithError(err, reqCtx.Log, "")
- }
- patch := client.MergeFrom(backupPolicy.DeepCopy())
- if cronJob.Status.LastScheduleTime != nil {
- backupPolicy.Status.LastScheduleTime = cronJob.Status.LastScheduleTime
- backupPolicy.Status.LastSuccessfulTime = cronJob.Status.LastSuccessfulTime
- if err := r.Client.Status().Patch(ctx, backupPolicy, patch); err != nil {
- return intctrlutil.CheckedRequeueWithError(err, reqCtx.Log, "")
- }
- }
- return intctrlutil.Reconciled()
-}
-
-// SetupWithManager sets up the controller with the Manager.
-func (r *CronJobReconciler) SetupWithManager(mgr ctrl.Manager) error {
- return ctrl.NewControllerManagedBy(mgr).
- For(&batchv1.CronJob{}).
- Owns(&batchv1.Job{}).
- WithEventFilter(predicate.NewPredicateFuncs(intctrlutil.ManagedByKubeBlocksFilterPredicate)).
- Complete(r)
-}
diff --git a/controllers/dataprotection/cue/cronjob.cue b/controllers/dataprotection/cue/cronjob.cue
deleted file mode 100644
index 830d0fd7d84..00000000000
--- a/controllers/dataprotection/cue/cronjob.cue
+++ /dev/null
@@ -1,138 +0,0 @@
-//Copyright (C) 2022-2023 ApeCloud Co., Ltd
-//
-//This file is part of KubeBlocks project
-//
-//This program is free software: you can redistribute it and/or modify
-//it under the terms of the GNU Affero General Public License as published by
-//the Free Software Foundation, either version 3 of the License, or
-//(at your option) any later version.
-//
-//This program is distributed in the hope that it will be useful
-//but WITHOUT ANY WARRANTY; without even the implied warranty of
-//MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
-//GNU Affero General Public License for more details.
-//
-//You should have received a copy of the GNU Affero General Public License
-//along with this program. If not, see .
-
-options: {
- name: string
- backupPolicyName: string
- namespace: string
- mgrNamespace: string
- cluster: string
- schedule: string
- backupType: string
- ttl: string
- serviceAccount: string
- image: string
- tolerations: {
- tolerations: [...]
- affinity: {...}
- nodeSelector: {...}
- }
-}
-
-cronjob: {
- apiVersion: "batch/v1"
- kind: "CronJob"
- metadata: {
- name: options.name
- namespace: options.mgrNamespace
- annotations:
- "kubeblocks.io/backup-namespace": options.namespace
- labels:
- "app.kubernetes.io/managed-by": "kubeblocks"
- }
- spec: {
- schedule: options.schedule
- successfulJobsHistoryLimit: 0
- failedJobsHistoryLimit: 1
- concurrencyPolicy: "Forbid"
- jobTemplate: spec: template: spec: {
- restartPolicy: "Never"
- serviceAccountName: options.serviceAccount
- affinity: options.tolerations.affinity
- tolerations: options.tolerations.tolerations
- nodeSelector: options.tolerations.nodeSelector
- containers: [{
- name: "backup-policy"
- image: options.image
- imagePullPolicy: "IfNotPresent"
- command: [
- "sh",
- "-c",
- ]
- args: [
- """
-kubectl create -f - <.
-
-options: {
- backupName: string
- containerName: string
- namespace: string
- image: string
- imagePullPolicy: string
-}
-
-container: {
- image: options.image
- name: options.containerName
- imagePullPolicy: options.imagePullPolicy
- command: ["sh", "-c"]
- args: [
- """
-retryTimes=0
-oldBackupInfo=
-trap "echo 'Terminating...' && exit" TERM
-while true; do
- sleep 3;
- if [ ! -f ${BACKUP_INFO_FILE} ]; then
- continue
- fi
- backupInfo=$(cat ${BACKUP_INFO_FILE})
- if [ "${oldBackupInfo}" == "${backupInfo}" ]; then
- continue
- fi
- echo "start to patch backupInfo: ${backupInfo}"
- eval kubectl -n \(options.namespace) patch backup \(options.backupName) --subresource=status --type=merge --patch '{\\\"status\\\":${backupInfo}}'
- if [ $? -ne 0 ]; then
- retryTimes=$(($retryTimes+1))
- else
- echo "update backup status successfully"
- retryTimes=0
- oldBackupInfo=${backupInfo}
- fi
- if [ $retryTimes -ge 3 ]; then
- echo "ERROR: update backup status failed, 3 attempts have been made!"
- exit 1
- fi
-done
-""",
- ]
-}
diff --git a/controllers/dataprotection/restore_controller.go b/controllers/dataprotection/restore_controller.go
new file mode 100644
index 00000000000..3e2b0f7d95e
--- /dev/null
+++ b/controllers/dataprotection/restore_controller.go
@@ -0,0 +1,389 @@
+/*
+Copyright (C) 2022-2023 ApeCloud Co., Ltd
+
+This file is part of KubeBlocks project
+
+This program is free software: you can redistribute it and/or modify
+it under the terms of the GNU Affero General Public License as published by
+the Free Software Foundation, either version 3 of the License, or
+(at your option) any later version.
+
+This program is distributed in the hope that it will be useful
+but WITHOUT ANY WARRANTY; without even the implied warranty of
+MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+GNU Affero General Public License for more details.
+
+You should have received a copy of the GNU Affero General Public License
+along with this program. If not, see .
+*/
+
+package dataprotection
+
+import (
+ "context"
+ "fmt"
+ "reflect"
+ "time"
+
+ batchv1 "k8s.io/api/batch/v1"
+ corev1 "k8s.io/api/core/v1"
+ "k8s.io/apimachinery/pkg/api/meta"
+ metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
+ "k8s.io/apimachinery/pkg/runtime"
+ "k8s.io/client-go/tools/record"
+ "k8s.io/klog/v2"
+ ctrl "sigs.k8s.io/controller-runtime"
+ "sigs.k8s.io/controller-runtime/pkg/client"
+ "sigs.k8s.io/controller-runtime/pkg/controller/controllerutil"
+ "sigs.k8s.io/controller-runtime/pkg/log"
+
+ dpv1alpha1 "github.com/apecloud/kubeblocks/apis/dataprotection/v1alpha1"
+ "github.com/apecloud/kubeblocks/internal/constant"
+ intctrlutil "github.com/apecloud/kubeblocks/internal/controllerutil"
+ dperrors "github.com/apecloud/kubeblocks/internal/dataprotection/errors"
+ dprestore "github.com/apecloud/kubeblocks/internal/dataprotection/restore"
+ dptypes "github.com/apecloud/kubeblocks/internal/dataprotection/types"
+)
+
+// RestoreReconciler reconciles a Restore object
+type RestoreReconciler struct {
+ client.Client
+ Scheme *runtime.Scheme
+ Recorder record.EventRecorder
+}
+
+// +kubebuilder:rbac:groups=dataprotection.kubeblocks.io,resources=restores,verbs=get;list;watch;create;update;patch;delete
+// +kubebuilder:rbac:groups=dataprotection.kubeblocks.io,resources=restores/status,verbs=get;update;patch
+// +kubebuilder:rbac:groups=dataprotection.kubeblocks.io,resources=restores/finalizers,verbs=update
+// +kubebuilder:rbac:groups=batch,resources=jobs,verbs=get;list;watch;create;update;patch;delete
+
+// Reconcile is part of the main kubernetes reconciliation loop which aims to
+// move the current state of the cluster closer to the desired state.
+//
+// For more details, check Reconcile and its Result here:
+// - https://pkg.go.dev/sigs.k8s.io/controller-runtime@v0.11.0/pkg/reconcile
+func (r *RestoreReconciler) Reconcile(ctx context.Context, req ctrl.Request) (ctrl.Result, error) {
+ reqCtx := intctrlutil.RequestCtx{
+ Ctx: ctx,
+ Req: req,
+ Log: log.FromContext(ctx).WithValues("backup", req.NamespacedName),
+ Recorder: r.Recorder,
+ }
+
+ // Get restore CR
+ restore := &dpv1alpha1.Restore{}
+ if err := r.Client.Get(reqCtx.Ctx, reqCtx.Req.NamespacedName, restore); err != nil {
+ return intctrlutil.CheckedRequeueWithError(err, reqCtx.Log, "")
+ }
+
+ // handle finalizer
+ res, err := intctrlutil.HandleCRDeletion(reqCtx, r, restore, dptypes.DataProtectionFinalizerName, func() (*ctrl.Result, error) {
+ return nil, r.deleteExternalResources(reqCtx, restore)
+ })
+ if res != nil {
+ return *res, err
+ }
+
+ switch restore.Status.Phase {
+ case "":
+ return r.newAction(reqCtx, restore)
+ case dpv1alpha1.RestorePhaseRunning:
+ return r.inProgressAction(reqCtx, restore)
+ }
+ return intctrlutil.Reconciled()
+}
+
+// SetupWithManager sets up the controller with the Manager.
+func (r *RestoreReconciler) SetupWithManager(mgr ctrl.Manager) error {
+ return ctrl.NewControllerManagedBy(mgr).
+ For(&dpv1alpha1.Restore{}).
+ Owns(&batchv1.Job{}).
+ Complete(r)
+}
+
+func (r *RestoreReconciler) deleteExternalResources(reqCtx intctrlutil.RequestCtx, restore *dpv1alpha1.Restore) error {
+ jobs := &batchv1.JobList{}
+ if err := r.Client.List(reqCtx.Ctx, jobs,
+ client.InNamespace(restore.Namespace),
+ client.MatchingLabels(dprestore.BuildRestoreLabels(restore.Name))); err != nil {
+ return client.IgnoreNotFound(err)
+ }
+ for i := range jobs.Items {
+ job := &jobs.Items[i]
+ if controllerutil.ContainsFinalizer(job, dptypes.DataProtectionFinalizerName) {
+ patch := client.MergeFrom(job.DeepCopy())
+ controllerutil.RemoveFinalizer(job, dptypes.DataProtectionFinalizerName)
+ if err := r.Patch(reqCtx.Ctx, job, patch); err != nil {
+ return err
+ }
+ }
+ }
+ return nil
+}
+
+func (r *RestoreReconciler) newAction(reqCtx intctrlutil.RequestCtx, restore *dpv1alpha1.Restore) (ctrl.Result, error) {
+ oldRestore := restore.DeepCopy()
+ patch := client.MergeFrom(oldRestore)
+ // patch metaObject
+ if restore.Labels == nil {
+ restore.Labels = map[string]string{}
+ }
+ restore.Labels[constant.AppManagedByLabelKey] = constant.AppName
+ if !reflect.DeepEqual(restore.ObjectMeta, oldRestore.ObjectMeta) {
+ if err := r.Client.Patch(reqCtx.Ctx, restore, patch); err != nil {
+ return intctrlutil.CheckedRequeueWithError(err, reqCtx.Log, "")
+ }
+ return intctrlutil.Reconciled()
+ }
+ // patch status
+ restore.Status.StartTimestamp = &metav1.Time{Time: time.Now()}
+ restore.Status.Phase = dpv1alpha1.RestorePhaseRunning
+ r.Recorder.Event(restore, corev1.EventTypeNormal, dprestore.ReasonRestoreStarting, "start to restore")
+ if err := r.Client.Status().Patch(reqCtx.Ctx, restore, patch); err != nil {
+ return intctrlutil.CheckedRequeueWithError(err, reqCtx.Log, "")
+ }
+ return intctrlutil.Reconciled()
+}
+
+func (r *RestoreReconciler) inProgressAction(reqCtx intctrlutil.RequestCtx, restore *dpv1alpha1.Restore) (ctrl.Result, error) {
+ restoreMgr := dprestore.NewRestoreManager(restore, r.Recorder, r.Scheme)
+ // handle restore actions
+ err := r.handleRestoreActions(reqCtx, restoreMgr)
+ if intctrlutil.IsTargetError(err, intctrlutil.ErrorTypeFatal) {
+ // set restore phase to failed if the error is fatal.
+ restoreMgr.Restore.Status.Phase = dpv1alpha1.RestorePhaseFailed
+ restoreMgr.Restore.Status.CompletionTimestamp = &metav1.Time{Time: time.Now()}
+ restoreMgr.Restore.Status.Duration = dprestore.GetRestoreDuration(restoreMgr.Restore.Status)
+ r.Recorder.Event(restore, corev1.EventTypeWarning, dprestore.ReasonRestoreFailed, err.Error())
+ err = nil
+ }
+ // patch restore status if changes occur
+ if !reflect.DeepEqual(restoreMgr.OriginalRestore.Status, restoreMgr.Restore.Status) {
+ err = r.Client.Status().Patch(reqCtx.Ctx, restoreMgr.Restore, client.MergeFrom(restoreMgr.OriginalRestore))
+ }
+ if err != nil {
+ return intctrlutil.RequeueWithError(err, reqCtx.Log, "")
+ }
+ return intctrlutil.Reconciled()
+}
+
+func (r *RestoreReconciler) handleRestoreActions(reqCtx intctrlutil.RequestCtx, restoreMgr *dprestore.RestoreManager) error {
+ // 1. validate if the restore.spec is valid and build restore manager.
+ if err := r.validateAndBuildMGR(reqCtx, restoreMgr); err != nil {
+ return err
+ }
+
+ // 2. handle the prepareData stage.
+ isCompleted, err := r.prepareData(reqCtx, restoreMgr)
+ if err != nil {
+ return err
+ }
+ // if prepareData is not completed, return
+ if !isCompleted {
+ return nil
+ }
+ // 3. handle the postReady stage.
+ isCompleted, err = r.postReady(reqCtx, restoreMgr)
+ if err != nil {
+ return err
+ }
+ if isCompleted {
+ restoreMgr.Restore.Status.Phase = dpv1alpha1.RestorePhaseCompleted
+ restoreMgr.Restore.Status.CompletionTimestamp = &metav1.Time{Time: time.Now()}
+ restoreMgr.Restore.Status.Duration = dprestore.GetRestoreDuration(restoreMgr.Restore.Status)
+ r.Recorder.Event(restoreMgr.Restore, corev1.EventTypeNormal, dprestore.ReasonRestoreCompleted, "restore completed.")
+ }
+ return nil
+}
+
+// validateAndBuildMGR validates the spec is valid to restore. if ok, build a manager for restoring.
+func (r *RestoreReconciler) validateAndBuildMGR(reqCtx intctrlutil.RequestCtx, restoreMgr *dprestore.RestoreManager) (err error) {
+ defer func() {
+ if err == nil {
+ dprestore.SetRestoreValidationCondition(restoreMgr.Restore, dprestore.ReasonValidateSuccessfully, "validate restore spec successfully")
+ } else if intctrlutil.IsTargetError(err, intctrlutil.ErrorTypeFatal) {
+ dprestore.SetRestoreValidationCondition(restoreMgr.Restore, dprestore.ReasonValidateFailed, err.Error())
+ r.Recorder.Event(restoreMgr.Restore, corev1.EventTypeWarning, dprestore.ReasonValidateFailed, err.Error())
+ }
+ }()
+
+ // get backupActionSet based on the specified backup name.
+ backupName := restoreMgr.Restore.Spec.Backup.Name
+ backupSet, err := restoreMgr.GetBackupActionSetByNamespaced(reqCtx, r.Client, backupName, restoreMgr.Restore.Spec.Backup.Namespace)
+ if err != nil {
+ return err
+ }
+
+ // check if the backup is completed exclude continuous backup.
+ var backupType dpv1alpha1.BackupType
+ if backupSet.ActionSet != nil {
+ backupType = backupSet.ActionSet.Spec.BackupType
+ } else if backupSet.UseVolumeSnapshot {
+ backupType = dpv1alpha1.BackupTypeFull
+ }
+ if backupType != dpv1alpha1.BackupTypeContinuous && backupSet.Backup.Status.Phase != dpv1alpha1.BackupPhaseCompleted {
+ err = intctrlutil.NewFatalError(fmt.Sprintf(`phase of backup "%s" is not completed`, backupName))
+ return err
+ }
+
+ // build backupActionSets of prepareData and postReady stage based on the specified backup's type.
+ switch backupType {
+ case dpv1alpha1.BackupTypeFull:
+ restoreMgr.SetBackupSets(*backupSet)
+ case dpv1alpha1.BackupTypeIncremental:
+ err = restoreMgr.BuildIncrementalBackupActionSets(reqCtx, r.Client, *backupSet)
+ case dpv1alpha1.BackupTypeDifferential:
+ err = restoreMgr.BuildDifferentialBackupActionSets(reqCtx, r.Client, *backupSet)
+ case dpv1alpha1.BackupTypeContinuous:
+ err = intctrlutil.NewErrorf(dperrors.ErrorTypeWaitForExternalHandler, "wait for external handler to do handle the Point-In-Time recovery.")
+ r.Recorder.Event(restoreMgr.Restore, corev1.EventTypeWarning, string(dperrors.ErrorTypeWaitForExternalHandler), err.Error())
+ default:
+ err = intctrlutil.NewFatalError(fmt.Sprintf("backup type of %s is empty", backupName))
+ }
+ return err
+}
+
+// prepareData handles the prepareData stage of the backups.
+func (r *RestoreReconciler) prepareData(reqCtx intctrlutil.RequestCtx, restoreMgr *dprestore.RestoreManager) (bool, error) {
+ if len(restoreMgr.PrepareDataBackupSets) == 0 {
+ return true, nil
+ }
+ prepareDataConfig := restoreMgr.Restore.Spec.PrepareDataConfig
+ if prepareDataConfig == nil || (prepareDataConfig.RestoreVolumeClaimsTemplate == nil && len(prepareDataConfig.RestoreVolumeClaims) == 0) {
+ return true, nil
+ }
+ if meta.IsStatusConditionTrue(restoreMgr.Restore.Status.Conditions, dprestore.ConditionTypeRestorePreparedData) {
+ return true, nil
+ }
+ var (
+ err error
+ isCompleted bool
+ )
+ defer func() {
+ r.handleRestoreStageError(restoreMgr.Restore, dpv1alpha1.PrepareData, err)
+ }()
+ // set processing prepare data condition
+ dprestore.SetRestoreStageCondition(restoreMgr.Restore, dpv1alpha1.PrepareData, dprestore.ReasonProcessing, "processing prepareData stage.")
+ for i, v := range restoreMgr.PrepareDataBackupSets {
+ isCompleted, err = r.handleBackupActionSet(reqCtx, restoreMgr, v, dpv1alpha1.PrepareData, i)
+ if err != nil {
+ return false, err
+ }
+ // waiting for restore jobs finished.
+ if !isCompleted {
+ return false, nil
+ }
+ }
+ // set prepare data successfully condition
+ dprestore.SetRestoreStageCondition(restoreMgr.Restore, dpv1alpha1.PrepareData, dprestore.ReasonSucceed, "prepare data successfully")
+ return true, nil
+}
+
+func (r *RestoreReconciler) postReady(reqCtx intctrlutil.RequestCtx, restoreMgr *dprestore.RestoreManager) (bool, error) {
+ readyConfig := restoreMgr.Restore.Spec.ReadyConfig
+ if len(restoreMgr.PostReadyBackupSets) == 0 || readyConfig == nil {
+ return true, nil
+ }
+ if meta.IsStatusConditionTrue(restoreMgr.Restore.Status.Conditions, dprestore.ConditionTypeRestorePostReady) {
+ return true, nil
+ }
+ dprestore.SetRestoreStageCondition(restoreMgr.Restore, dpv1alpha1.PostReady, dprestore.ReasonProcessing, "processing postReady stage")
+ var (
+ err error
+ isCompleted bool
+ )
+ defer func() {
+ r.handleRestoreStageError(restoreMgr.Restore, dpv1alpha1.PrepareData, err)
+ }()
+ if readyConfig.ReadinessProbe != nil && !meta.IsStatusConditionTrue(restoreMgr.Restore.Status.Conditions, dprestore.ConditionTypeReadinessProbe) {
+ // TODO: check readiness probe, use a job or exec?
+ _ = klog.TODO()
+ }
+ for _, v := range restoreMgr.PostReadyBackupSets {
+ // handle postReady actions
+ for i := range v.ActionSet.Spec.Restore.PostReady {
+ isCompleted, err = r.handleBackupActionSet(reqCtx, restoreMgr, v, dpv1alpha1.PostReady, i)
+ if err != nil {
+ return false, err
+ }
+ // waiting for restore jobs finished.
+ if !isCompleted {
+ return false, nil
+ }
+ }
+ }
+ dprestore.SetRestoreStageCondition(restoreMgr.Restore, dpv1alpha1.PostReady, dprestore.ReasonSucceed, "processing postReady stage successfully")
+ return true, nil
+}
+
+func (r *RestoreReconciler) handleBackupActionSet(reqCtx intctrlutil.RequestCtx,
+ restoreMgr *dprestore.RestoreManager,
+ backupSet dprestore.BackupActionSet,
+ stage dpv1alpha1.RestoreStage,
+ step int) (bool, error) {
+ handleFailed := func(restore *dpv1alpha1.Restore, backupName string) error {
+ errorMsg := fmt.Sprintf(`restore failed for backup "%s", more information can be found in status.actions.%s`, backupName, stage)
+ dprestore.SetRestoreStageCondition(restore, stage, dprestore.ReasonFailed, errorMsg)
+ return intctrlutil.NewFatalError(errorMsg)
+ }
+
+ checkIsCompleted := func(allActionsFinished, existFailedAction bool) (bool, error) {
+ if !allActionsFinished {
+ return false, nil
+ }
+ if existFailedAction {
+ return true, handleFailed(restoreMgr.Restore, backupSet.Backup.Name)
+ }
+ return true, nil
+ }
+
+ actionName := fmt.Sprintf("%s-%d", stage, step)
+ // 1. check if the restore actions are completed from status.actions firstly.
+ allActionsFinished, existFailedAction := restoreMgr.AnalysisRestoreActionsWithBackup(stage, backupSet.Backup.Name, actionName)
+ isCompleted, err := checkIsCompleted(allActionsFinished, existFailedAction)
+ if isCompleted || err != nil {
+ return isCompleted, err
+ }
+
+ var jobs []*batchv1.Job
+ switch stage {
+ case dpv1alpha1.PrepareData:
+ if backupSet.UseVolumeSnapshot {
+ if err = restoreMgr.RestorePVCFromSnapshot(reqCtx, r.Client, backupSet, actionName); err != nil {
+ return false, nil
+ }
+ }
+ jobs, err = restoreMgr.BuildPrepareDataJobs(reqCtx, r.Client, backupSet, actionName)
+ case dpv1alpha1.PostReady:
+ // 2. build jobs for postReady action
+ jobs, err = restoreMgr.BuildPostReadyActionJobs(reqCtx, r.Client, backupSet, backupSet.ActionSet.Spec.Restore.PostReady[step])
+ }
+ if err != nil {
+ return false, err
+ }
+ if len(jobs) == 0 {
+ return true, nil
+ }
+ // 3. create jobs
+ jobs, err = restoreMgr.CreateJobsIfNotExist(reqCtx, r.Client, jobs)
+ if err != nil {
+ return false, err
+ }
+
+ // 4. check if jobs are finished.
+ allActionsFinished, existFailedAction = restoreMgr.CheckJobsDone(stage, actionName, backupSet, jobs)
+ if stage == dpv1alpha1.PrepareData {
+ // recalculation whether all actions have been completed.
+ restoreMgr.Recalculation(backupSet.Backup.Name, actionName, &allActionsFinished, &existFailedAction)
+ }
+ return checkIsCompleted(allActionsFinished, existFailedAction)
+}
+
+func (r *RestoreReconciler) handleRestoreStageError(restore *dpv1alpha1.Restore, stage dpv1alpha1.RestoreStage, err error) {
+ if intctrlutil.IsTargetError(err, intctrlutil.ErrorTypeFatal) {
+ condition := meta.FindStatusCondition(restore.Status.Conditions, dprestore.ConditionTypeRestorePreparedData)
+ if condition != nil && condition.Reason != dprestore.ReasonFailed {
+ dprestore.SetRestoreStageCondition(restore, stage, dprestore.ReasonFailed, err.Error())
+ }
+ }
+}
diff --git a/controllers/dataprotection/restorejob_controller.go b/controllers/dataprotection/restorejob_controller.go
deleted file mode 100644
index e38ab27351e..00000000000
--- a/controllers/dataprotection/restorejob_controller.go
+++ /dev/null
@@ -1,350 +0,0 @@
-/*
-Copyright (C) 2022-2023 ApeCloud Co., Ltd
-
-This file is part of KubeBlocks project
-
-This program is free software: you can redistribute it and/or modify
-it under the terms of the GNU Affero General Public License as published by
-the Free Software Foundation, either version 3 of the License, or
-(at your option) any later version.
-
-This program is distributed in the hope that it will be useful
-but WITHOUT ANY WARRANTY; without even the implied warranty of
-MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
-GNU Affero General Public License for more details.
-
-You should have received a copy of the GNU Affero General Public License
-along with this program. If not, see .
-*/
-
-package dataprotection
-
-import (
- "context"
- "fmt"
-
- appv1 "k8s.io/api/apps/v1"
- batchv1 "k8s.io/api/batch/v1"
- corev1 "k8s.io/api/core/v1"
- metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
- "k8s.io/apimachinery/pkg/runtime"
- "k8s.io/apimachinery/pkg/types"
- "k8s.io/client-go/tools/record"
- "k8s.io/utils/clock"
- ctrl "sigs.k8s.io/controller-runtime"
- "sigs.k8s.io/controller-runtime/pkg/client"
- "sigs.k8s.io/controller-runtime/pkg/controller"
- "sigs.k8s.io/controller-runtime/pkg/log"
-
- dataprotectionv1alpha1 "github.com/apecloud/kubeblocks/apis/dataprotection/v1alpha1"
- "github.com/apecloud/kubeblocks/internal/constant"
- intctrlutil "github.com/apecloud/kubeblocks/internal/controllerutil"
- viper "github.com/apecloud/kubeblocks/internal/viperx"
-)
-
-// RestoreJobReconciler reconciles a RestoreJob object
-type RestoreJobReconciler struct {
- client.Client
- Scheme *runtime.Scheme
- Recorder record.EventRecorder
- clock clock.RealClock
-}
-
-// +kubebuilder:rbac:groups=dataprotection.kubeblocks.io,resources=restorejobs,verbs=get;list;watch;create;update;patch;delete
-// +kubebuilder:rbac:groups=dataprotection.kubeblocks.io,resources=restorejobs/status,verbs=get;update;patch
-// +kubebuilder:rbac:groups=dataprotection.kubeblocks.io,resources=restorejobs/finalizers,verbs=update
-
-// Reconcile is part of the main kubernetes reconciliation loop which aims to
-// move the current state of the cluster closer to the desired state.
-// TODO(user): Modify the Reconcile function to compare the state specified by
-// the RestoreJob object against the actual cluster state, and then
-// perform operations to make the cluster state reflect the state specified by
-// the user.
-//
-// For more details, check Reconcile and its Result here:
-// - https://pkg.go.dev/sigs.k8s.io/controller-runtime@v0.12.1/pkg/reconcile
-func (r *RestoreJobReconciler) Reconcile(ctx context.Context, req ctrl.Request) (ctrl.Result, error) {
- // NOTES:
- // setup common request context
- reqCtx := intctrlutil.RequestCtx{
- Ctx: ctx,
- Req: req,
- Log: log.FromContext(ctx).WithValues("restoreJob", req.NamespacedName),
- Recorder: r.Recorder,
- }
- restoreJob := &dataprotectionv1alpha1.RestoreJob{}
- if err := r.Client.Get(reqCtx.Ctx, reqCtx.Req.NamespacedName, restoreJob); err != nil {
- return intctrlutil.CheckedRequeueWithError(err, reqCtx.Log, "")
- }
- reqCtx.Log.Info("in RestoreJob Reconciler: name: " + restoreJob.Name + " phase: " + string(restoreJob.Status.Phase))
-
- // handle finalizer
- res, err := intctrlutil.HandleCRDeletion(reqCtx, r, restoreJob, dataProtectionFinalizerName, func() (*ctrl.Result, error) {
- return nil, r.deleteExternalResources(reqCtx, restoreJob)
- })
- if res != nil {
- return *res, err
- }
-
- switch restoreJob.Status.Phase {
- case "", dataprotectionv1alpha1.RestoreJobNew:
- return r.doRestoreNewPhaseAction(reqCtx, restoreJob)
- case dataprotectionv1alpha1.RestoreJobInProgressPhy:
- return r.doRestoreInProgressPhyAction(reqCtx, restoreJob)
- default:
- return intctrlutil.Reconciled()
- }
-}
-
-// SetupWithManager sets up the controller with the Manager.
-func (r *RestoreJobReconciler) SetupWithManager(mgr ctrl.Manager) error {
- return ctrl.NewControllerManagedBy(mgr).
- For(&dataprotectionv1alpha1.RestoreJob{}).
- WithOptions(controller.Options{
- MaxConcurrentReconciles: viper.GetInt(maxConcurDataProtectionReconKey),
- }).
- Complete(r)
-}
-
-func (r *RestoreJobReconciler) doRestoreNewPhaseAction(
- reqCtx intctrlutil.RequestCtx,
- restoreJob *dataprotectionv1alpha1.RestoreJob) (ctrl.Result, error) {
-
- // 1. get stateful service and
- // 2. set stateful replicas to 0
- patch := []byte(`{"spec":{"replicas":0}}`)
- if err := r.patchTargetCluster(reqCtx, restoreJob, patch); err != nil {
- return intctrlutil.CheckedRequeueWithError(err, reqCtx.Log, "")
- }
-
- // get backup tool
- // get backup job
- // build a job pod sec
- jobPodSpec, err := r.buildPodSpec(reqCtx, restoreJob)
- if err != nil {
- return intctrlutil.CheckedRequeueWithError(err, reqCtx.Log, "")
- }
-
- job := &batchv1.Job{
- ObjectMeta: metav1.ObjectMeta{
- Namespace: restoreJob.Namespace,
- Name: restoreJob.Name,
- Labels: buildRestoreJobLabels(restoreJob.Name),
- },
- Spec: batchv1.JobSpec{
- Template: corev1.PodTemplateSpec{
- ObjectMeta: metav1.ObjectMeta{
- Namespace: restoreJob.Namespace,
- Name: restoreJob.Name},
- Spec: jobPodSpec,
- },
- },
- }
- reqCtx.Log.Info("create a built-in job from restoreJob", "job", job)
-
- if err := r.Client.Create(reqCtx.Ctx, job); err != nil {
- return intctrlutil.CheckedRequeueWithError(err, reqCtx.Log, "")
- }
-
- // update Phase to InProgress
- restoreJob.Status.Phase = dataprotectionv1alpha1.RestoreJobInProgressPhy
- restoreJob.Status.StartTimestamp = &metav1.Time{Time: r.clock.Now().UTC()}
- if err := r.Client.Status().Update(reqCtx.Ctx, restoreJob); err != nil {
- return intctrlutil.CheckedRequeueWithError(err, reqCtx.Log, "")
- }
- return intctrlutil.Reconciled()
-}
-
-func (r *RestoreJobReconciler) doRestoreInProgressPhyAction(
- reqCtx intctrlutil.RequestCtx,
- restoreJob *dataprotectionv1alpha1.RestoreJob) (ctrl.Result, error) {
- job, err := r.getBatchV1Job(reqCtx, restoreJob)
- if err != nil {
- // not found backup job, retry create job
- reqCtx.Log.Info(err.Error())
- return intctrlutil.CheckedRequeueWithError(err, reqCtx.Log, "")
- }
- jobStatusConditions := job.Status.Conditions
- if len(jobStatusConditions) == 0 {
- return intctrlutil.RequeueAfter(reconcileInterval, reqCtx.Log, "")
- }
-
- switch jobStatusConditions[0].Type {
- case batchv1.JobComplete:
- // update Phase to Completed
- restoreJob.Status.Phase = dataprotectionv1alpha1.RestoreJobCompleted
- restoreJob.Status.CompletionTimestamp = &metav1.Time{Time: r.clock.Now().UTC()}
- // get stateful service and
- // set stateful replicas to 1
- patch := []byte(`{"spec":{"replicas":1}}`)
- if err := r.patchTargetCluster(reqCtx, restoreJob, patch); err != nil {
- return intctrlutil.CheckedRequeueWithError(err, reqCtx.Log, "")
- }
- case batchv1.JobFailed:
- restoreJob.Status.Phase = dataprotectionv1alpha1.RestoreJobFailed
- restoreJob.Status.FailureReason = job.Status.Conditions[0].Reason
- }
- if err := r.Client.Status().Update(reqCtx.Ctx, restoreJob); err != nil {
- return intctrlutil.CheckedRequeueWithError(err, reqCtx.Log, "")
- }
- return intctrlutil.Reconciled()
-}
-
-func (r *RestoreJobReconciler) deleteExternalResources(reqCtx intctrlutil.RequestCtx, restoreJob *dataprotectionv1alpha1.RestoreJob) error {
- //
- // delete any external resources associated with the cronJob
- //
- // Ensure that delete implementation is idempotent and safe to invoke
- // multiple times for same object.
-
- // delete k8s job.
- job, err := r.getBatchV1Job(reqCtx, restoreJob)
- if err != nil {
- // not found backup job, do nothing
- reqCtx.Log.Info(err.Error())
- return nil
- }
-
- if err := intctrlutil.BackgroundDeleteObject(r.Client, reqCtx.Ctx, job); err != nil {
- return err
- }
- return nil
-}
-
-func (r *RestoreJobReconciler) getBatchV1Job(reqCtx intctrlutil.RequestCtx, backup *dataprotectionv1alpha1.RestoreJob) (*batchv1.Job, error) {
- job := &batchv1.Job{}
- jobNameSpaceName := types.NamespacedName{
- Namespace: reqCtx.Req.Namespace,
- Name: backup.Name,
- }
- if err := r.Client.Get(reqCtx.Ctx, jobNameSpaceName, job); err != nil {
- // not found backup job, do nothing
- reqCtx.Log.Info(err.Error())
- return nil, err
- }
- return job, nil
-}
-
-func (r *RestoreJobReconciler) buildPodSpec(reqCtx intctrlutil.RequestCtx, restoreJob *dataprotectionv1alpha1.RestoreJob) (corev1.PodSpec, error) {
- var podSpec corev1.PodSpec
- logger := reqCtx.Log
-
- // get backup job
- backup := &dataprotectionv1alpha1.Backup{}
- backupNameSpaceName := types.NamespacedName{
- Namespace: reqCtx.Req.Namespace,
- Name: restoreJob.Spec.BackupJobName,
- }
- if err := r.Get(reqCtx.Ctx, backupNameSpaceName, backup); err != nil {
- logger.Error(err, "Unable to get backup for restore.", "backup", backupNameSpaceName)
- return podSpec, err
- }
-
- // get backup tool
- backupTool := &dataprotectionv1alpha1.BackupTool{}
- backupToolNameSpaceName := types.NamespacedName{
- Namespace: reqCtx.Req.Namespace,
- Name: backup.Status.BackupToolName,
- }
- if err := r.Client.Get(reqCtx.Ctx, backupToolNameSpaceName, backupTool); err != nil {
- logger.Error(err, "Unable to get backupTool for backup.", "BackupTool", backupToolNameSpaceName)
- return podSpec, err
- }
-
- if len(backup.Status.PersistentVolumeClaimName) == 0 {
- return podSpec, nil
- }
-
- container := corev1.Container{}
- container.Name = restoreJob.Name
- container.Command = []string{"sh", "-c"}
- container.Args = backupTool.Spec.Physical.GetPhysicalRestoreCommand()
- container.Image = backupTool.Spec.Image
- if backupTool.Spec.Resources != nil {
- container.Resources = *backupTool.Spec.Resources
- }
-
- container.VolumeMounts = restoreJob.Spec.TargetVolumeMounts
-
- // add the volumeMounts with backup volume
- restoreVolumeName := fmt.Sprintf("restore-%s", backup.Status.PersistentVolumeClaimName)
- remoteVolume := corev1.Volume{
- Name: restoreVolumeName,
- VolumeSource: corev1.VolumeSource{
- PersistentVolumeClaim: &corev1.PersistentVolumeClaimVolumeSource{
- ClaimName: backup.Status.PersistentVolumeClaimName,
- },
- },
- }
- // add remote volumeMounts
- remoteVolumeMount := corev1.VolumeMount{}
- remoteVolumeMount.Name = restoreVolumeName
- remoteVolumeMount.MountPath = "/data"
- container.VolumeMounts = append(container.VolumeMounts, remoteVolumeMount)
-
- allowPrivilegeEscalation := false
- runAsUser := int64(0)
- container.SecurityContext = &corev1.SecurityContext{
- AllowPrivilegeEscalation: &allowPrivilegeEscalation,
- RunAsUser: &runAsUser}
-
- // build env for restore
- envBackupName := corev1.EnvVar{
- Name: "BACKUP_NAME",
- Value: backup.Name,
- }
-
- container.Env = []corev1.EnvVar{envBackupName}
- // merge env from backup tool.
- container.Env = append(container.Env, backupTool.Spec.Env...)
-
- podSpec.Containers = []corev1.Container{container}
-
- podSpec.Volumes = restoreJob.Spec.TargetVolumes
-
- // add remote volumes
- podSpec.Volumes = append(podSpec.Volumes, remoteVolume)
-
- // TODO(dsj): mount readonly remote volumes for restore.
- // podSpec.Volumes[0].PersistentVolumeClaim.ReadOnly = true
- podSpec.RestartPolicy = corev1.RestartPolicyNever
-
- return podSpec, nil
-}
-
-func (r *RestoreJobReconciler) patchTargetCluster(reqCtx intctrlutil.RequestCtx, restoreJob *dataprotectionv1alpha1.RestoreJob, patch []byte) error {
- // get stateful service
- clusterTarget := &appv1.StatefulSetList{}
- if err := r.Client.List(reqCtx.Ctx, clusterTarget,
- client.InNamespace(reqCtx.Req.Namespace),
- client.MatchingLabels(restoreJob.Spec.Target.LabelsSelector.MatchLabels)); err != nil {
- return err
- }
- reqCtx.Log.Info("Get cluster target finish", "target", clusterTarget)
- clusterItemsLen := len(clusterTarget.Items)
- if clusterItemsLen != 1 {
- if clusterItemsLen <= 0 {
- restoreJob.Status.FailureReason = "Can not find any statefulsets with labelsSelector."
- } else {
- restoreJob.Status.FailureReason = "Match more than one results, please check the labelsSelector."
- }
- restoreJob.Status.Phase = dataprotectionv1alpha1.RestoreJobFailed
- reqCtx.Log.Info(restoreJob.Status.FailureReason)
- if err := r.Client.Status().Update(reqCtx.Ctx, restoreJob); err != nil {
- return err
- }
- return nil
- }
- // patch stateful set
- if err := r.Client.Patch(reqCtx.Ctx, &clusterTarget.Items[0], client.RawPatch(types.StrategicMergePatchType, patch)); err != nil {
- return err
- }
- return nil
-}
-
-func buildRestoreJobLabels(jobName string) map[string]string {
- return map[string]string{
- dataProtectionLabelRestoreJobNameKey: jobName,
- constant.AppManagedByLabelKey: constant.AppName,
- }
-}
diff --git a/controllers/dataprotection/restorejob_controller_test.go b/controllers/dataprotection/restorejob_controller_test.go
deleted file mode 100644
index 733d50d68d3..00000000000
--- a/controllers/dataprotection/restorejob_controller_test.go
+++ /dev/null
@@ -1,200 +0,0 @@
-/*
-Copyright (C) 2022-2023 ApeCloud Co., Ltd
-
-This file is part of KubeBlocks project
-
-This program is free software: you can redistribute it and/or modify
-it under the terms of the GNU Affero General Public License as published by
-the Free Software Foundation, either version 3 of the License, or
-(at your option) any later version.
-
-This program is distributed in the hope that it will be useful
-but WITHOUT ANY WARRANTY; without even the implied warranty of
-MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
-GNU Affero General Public License for more details.
-
-You should have received a copy of the GNU Affero General Public License
-along with this program. If not, see .
-*/
-
-package dataprotection
-
-import (
- . "github.com/onsi/ginkgo/v2"
- . "github.com/onsi/gomega"
-
- appsv1 "k8s.io/api/apps/v1"
- batchv1 "k8s.io/api/batch/v1"
- corev1 "k8s.io/api/core/v1"
- metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
- "k8s.io/apimachinery/pkg/types"
- "sigs.k8s.io/controller-runtime/pkg/client"
-
- dataprotectionv1alpha1 "github.com/apecloud/kubeblocks/apis/dataprotection/v1alpha1"
- "github.com/apecloud/kubeblocks/internal/constant"
- intctrlutil "github.com/apecloud/kubeblocks/internal/generics"
- testapps "github.com/apecloud/kubeblocks/internal/testutil/apps"
-)
-
-var _ = Describe("RestoreJob Controller", func() {
- const (
- clusterName = "mycluster"
- compName = "cluster"
- )
- cleanEnv := func() {
- // must wait till resources deleted and no longer existed before the testcases start,
- // otherwise if later it needs to create some new resource objects with the same name,
- // in race conditions, it will find the existence of old objects, resulting failure to
- // create the new objects.
- By("clean resources")
-
- // delete rest mocked objects
- inNS := client.InNamespace(testCtx.DefaultNamespace)
- ml := client.HasLabels{testCtx.TestObjLabelKey}
- // namespaced
- testapps.ClearResources(&testCtx, intctrlutil.StatefulSetSignature, inNS, ml)
- testapps.ClearResources(&testCtx, intctrlutil.PodSignature, inNS, ml)
- testapps.ClearResources(&testCtx, intctrlutil.RestoreJobSignature, inNS, ml)
- testapps.ClearResourcesWithRemoveFinalizerOption(&testCtx, intctrlutil.BackupSignature, true, inNS)
- testapps.ClearResources(&testCtx, intctrlutil.BackupPolicySignature, inNS, ml)
- testapps.ClearResourcesWithRemoveFinalizerOption(&testCtx, intctrlutil.JobSignature, true, inNS)
- testapps.ClearResources(&testCtx, intctrlutil.CronJobSignature, inNS, ml)
- // non-namespaced
- testapps.ClearResources(&testCtx, intctrlutil.BackupToolSignature, ml)
- testapps.ClearResources(&testCtx, intctrlutil.BackupPolicyTemplateSignature, ml)
- }
-
- BeforeEach(cleanEnv)
-
- AfterEach(cleanEnv)
-
- assureRestoreJobObj := func(backup string) *dataprotectionv1alpha1.RestoreJob {
- By("By assure an restoreJob obj")
- return testapps.NewRestoreJobFactory(testCtx.DefaultNamespace, "restore-job-").
- WithRandomName().SetBackupJobName(backup).
- SetTargetSecretName("mycluster-cluster-secret").
- AddTargetVolumePVC("mysql-restore-storage", "datadir-mycluster-0").
- AddTargetVolumeMount(corev1.VolumeMount{Name: "mysql-restore-storage", MountPath: "/var/lib/mysql"}).
- Create(&testCtx).GetObject()
- }
-
- assureBackupObj := func(backupPolicy string) *dataprotectionv1alpha1.Backup {
- By("By assure an backup obj")
- return testapps.NewBackupFactory(testCtx.DefaultNamespace, "backup-job-").
- WithRandomName().SetBackupPolicyName(backupPolicy).
- SetBackupType(dataprotectionv1alpha1.BackupTypeDataFile).
- Create(&testCtx).GetObject()
- }
-
- assureBackupPolicyObj := func(backupTool string) *dataprotectionv1alpha1.BackupPolicy {
- By("By assure an backupPolicy obj")
- return testapps.NewBackupPolicyFactory(testCtx.DefaultNamespace, "backup-policy-").
- WithRandomName().
- AddDataFilePolicy().
- AddMatchLabels(constant.AppInstanceLabelKey, clusterName).
- SetSchedule("0 3 * * *", true).
- SetTTL("7d").
- SetBackupToolName(backupTool).
- SetTargetSecretName("mycluster-cluster-secret").
- SetPVC("backup-host-path-pvc").
- Create(&testCtx).GetObject()
- }
-
- assureBackupToolObj := func(withoutResources ...bool) *dataprotectionv1alpha1.BackupTool {
- By("By assure an backupTool obj")
- return testapps.CreateCustomizedObj(&testCtx, "backup/backuptool.yaml",
- &dataprotectionv1alpha1.BackupTool{}, testapps.RandomizedObjName(),
- func(bt *dataprotectionv1alpha1.BackupTool) {
- nilResources := false
- // optional arguments, only use the first one.
- if len(withoutResources) > 0 {
- nilResources = withoutResources[0]
- }
- if nilResources {
- bt.Spec.Resources = nil
- }
- })
- }
-
- assureStatefulSetObj := func() *appsv1.StatefulSet {
- By("By assure an stateful obj")
- return testapps.NewStatefulSetFactory(testCtx.DefaultNamespace, clusterName, clusterName, compName).
- SetReplicas(3).
- AddAppInstanceLabel(clusterName).
- AddContainer(corev1.Container{Name: "mysql", Image: testapps.ApeCloudMySQLImage}).
- AddVolumeClaimTemplate(corev1.PersistentVolumeClaim{
- ObjectMeta: metav1.ObjectMeta{Name: testapps.DataVolumeName},
- Spec: testapps.NewPVC("1Gi"),
- }).Create(&testCtx).GetObject()
- }
-
- patchBackupStatus := func(phase dataprotectionv1alpha1.BackupPhase, key types.NamespacedName) {
- Eventually(testapps.GetAndChangeObjStatus(&testCtx, key, func(backup *dataprotectionv1alpha1.Backup) {
- backup.Status.Phase = phase
- })).Should(Succeed())
- }
-
- patchK8sJobStatus := func(jobStatus batchv1.JobConditionType, key types.NamespacedName) {
- Eventually(testapps.GetAndChangeObjStatus(&testCtx, key, func(job *batchv1.Job) {
- found := false
- for _, cond := range job.Status.Conditions {
- if cond.Type == jobStatus {
- found = true
- }
- }
- if !found {
- jobCondition := batchv1.JobCondition{Type: jobStatus}
- job.Status.Conditions = append(job.Status.Conditions, jobCondition)
- }
- })).Should(Succeed())
- }
-
- testRestoreJob := func(withResources ...bool) {
- By("By creating a statefulset and pod")
- sts := assureStatefulSetObj()
- testapps.MockConsensusComponentPods(&testCtx, sts, clusterName, compName)
-
- By("By creating a backupTool")
- backupTool := assureBackupToolObj(withResources...)
-
- By("By creating a backupPolicy from backupTool: " + backupTool.Name)
- backupPolicy := assureBackupPolicyObj(backupTool.Name)
-
- By("By creating a backup from backupPolicy: " + backupPolicy.Name)
- backup := assureBackupObj(backupPolicy.Name)
-
- By("By creating a restoreJob from backup: " + backup.Name)
- toCreate := assureRestoreJobObj(backup.Name)
- key := types.NamespacedName{
- Name: toCreate.Name,
- Namespace: toCreate.Namespace,
- }
- backupKey := types.NamespacedName{Name: backup.Name, Namespace: backup.Namespace}
- Eventually(testapps.CheckObj(&testCtx, backupKey, func(g Gomega, fetched *dataprotectionv1alpha1.Backup) {
- g.Expect(fetched.Status.Phase).To(Equal(dataprotectionv1alpha1.BackupInProgress))
- })).Should(Succeed())
-
- patchBackupStatus(dataprotectionv1alpha1.BackupCompleted, backupKey)
-
- patchK8sJobStatus(batchv1.JobComplete, key)
-
- result := &dataprotectionv1alpha1.RestoreJob{}
- Eventually(func() bool {
- Expect(k8sClient.Get(ctx, key, result)).Should(Succeed())
- return result.Status.Phase == dataprotectionv1alpha1.RestoreJobCompleted ||
- result.Status.Phase == dataprotectionv1alpha1.RestoreJobFailed
- }).Should(BeTrue())
- Expect(result.Status.Phase).Should(Equal(dataprotectionv1alpha1.RestoreJobCompleted))
- }
-
- Context("When creating restoreJob", func() {
- It("Should success with no error", func() {
- testRestoreJob()
- })
-
- It("Without backupTool resources should success with no error", func() {
- testRestoreJob(true)
- })
- })
-
-})
diff --git a/controllers/dataprotection/suite_test.go b/controllers/dataprotection/suite_test.go
index e060fa5a1f6..c8175186368 100644
--- a/controllers/dataprotection/suite_test.go
+++ b/controllers/dataprotection/suite_test.go
@@ -30,8 +30,8 @@ import (
. "github.com/onsi/ginkgo/v2"
. "github.com/onsi/gomega"
- snapshotv1beta1 "github.com/kubernetes-csi/external-snapshotter/client/v3/apis/volumesnapshot/v1beta1"
- snapshotv1 "github.com/kubernetes-csi/external-snapshotter/client/v6/apis/volumesnapshot/v1"
+ vsv1beta1 "github.com/kubernetes-csi/external-snapshotter/client/v3/apis/volumesnapshot/v1beta1"
+ vsv1 "github.com/kubernetes-csi/external-snapshotter/client/v6/apis/volumesnapshot/v1"
"go.uber.org/zap/zapcore"
batchv1 "k8s.io/api/batch/v1"
"k8s.io/client-go/kubernetes/scheme"
@@ -43,7 +43,7 @@ import (
"sigs.k8s.io/controller-runtime/pkg/log/zap"
appsv1alpha1 "github.com/apecloud/kubeblocks/apis/apps/v1alpha1"
- dataprotectionv1alpha1 "github.com/apecloud/kubeblocks/apis/dataprotection/v1alpha1"
+ dpv1alpha1 "github.com/apecloud/kubeblocks/apis/dataprotection/v1alpha1"
storagev1alpha1 "github.com/apecloud/kubeblocks/apis/storage/v1alpha1"
"github.com/apecloud/kubeblocks/internal/constant"
"github.com/apecloud/kubeblocks/internal/testutil"
@@ -103,16 +103,16 @@ var _ = BeforeSuite(func() {
scheme := scheme.Scheme
- err = snapshotv1.AddToScheme(scheme)
+ err = vsv1.AddToScheme(scheme)
Expect(err).NotTo(HaveOccurred())
- err = snapshotv1beta1.AddToScheme(scheme)
+ err = vsv1beta1.AddToScheme(scheme)
Expect(err).NotTo(HaveOccurred())
err = appsv1alpha1.AddToScheme(scheme)
Expect(err).NotTo(HaveOccurred())
- err = dataprotectionv1alpha1.AddToScheme(scheme)
+ err = dpv1alpha1.AddToScheme(scheme)
Expect(err).NotTo(HaveOccurred())
err = storagev1alpha1.AddToScheme(scheme)
@@ -125,13 +125,16 @@ var _ = BeforeSuite(func() {
Expect(k8sClient).NotTo(BeNil())
uncachedObjects := []client.Object{
- &dataprotectionv1alpha1.BackupPolicy{},
- &dataprotectionv1alpha1.BackupTool{},
- &dataprotectionv1alpha1.Backup{},
- &dataprotectionv1alpha1.RestoreJob{},
- &snapshotv1.VolumeSnapshot{},
- &snapshotv1beta1.VolumeSnapshot{},
+ &dpv1alpha1.ActionSet{},
+ &dpv1alpha1.BackupPolicy{},
+ &dpv1alpha1.BackupSchedule{},
+ &dpv1alpha1.BackupRepo{},
+ &dpv1alpha1.Backup{},
+ &dpv1alpha1.Restore{},
+ &vsv1.VolumeSnapshot{},
+ &vsv1beta1.VolumeSnapshot{},
&batchv1.Job{},
+ &batchv1.CronJob{},
}
// run reconcile
k8sManager, err := ctrl.NewManager(cfg, ctrl.Options{
@@ -147,33 +150,25 @@ var _ = BeforeSuite(func() {
Recorder: k8sManager.GetEventRecorderFor("backup-controller"),
}).SetupWithManager(k8sManager)
Expect(err).ToNot(HaveOccurred())
- Expect(err).ToNot(HaveOccurred())
-
- err = (&BackupPolicyReconciler{
- Client: k8sManager.GetClient(),
- Scheme: k8sManager.GetScheme(),
- Recorder: k8sManager.GetEventRecorderFor("backup-policy-controller"),
- }).SetupWithManager(k8sManager)
- Expect(err).ToNot(HaveOccurred())
- err = (&BackupToolReconciler{
+ err = (&BackupScheduleReconciler{
Client: k8sManager.GetClient(),
Scheme: k8sManager.GetScheme(),
- Recorder: k8sManager.GetEventRecorderFor("backup-tool-controller"),
+ Recorder: k8sManager.GetEventRecorderFor("backup-schedule-controller"),
}).SetupWithManager(k8sManager)
Expect(err).ToNot(HaveOccurred())
- err = (&RestoreJobReconciler{
+ err = (&BackupPolicyReconciler{
Client: k8sManager.GetClient(),
Scheme: k8sManager.GetScheme(),
- Recorder: k8sManager.GetEventRecorderFor("restore-job-controller"),
+ Recorder: k8sManager.GetEventRecorderFor("backup-policy-controller"),
}).SetupWithManager(k8sManager)
Expect(err).ToNot(HaveOccurred())
- err = (&CronJobReconciler{
+ err = (&ActionSetReconciler{
Client: k8sManager.GetClient(),
Scheme: k8sManager.GetScheme(),
- Recorder: k8sManager.GetEventRecorderFor("cronjob-controller"),
+ Recorder: k8sManager.GetEventRecorderFor("actionset-controller"),
}).SetupWithManager(k8sManager)
Expect(err).ToNot(HaveOccurred())
diff --git a/controllers/dataprotection/type.go b/controllers/dataprotection/type.go
deleted file mode 100644
index f2e4a09cda8..00000000000
--- a/controllers/dataprotection/type.go
+++ /dev/null
@@ -1,110 +0,0 @@
-/*
-Copyright (C) 2022-2023 ApeCloud Co., Ltd
-
-This file is part of KubeBlocks project
-
-This program is free software: you can redistribute it and/or modify
-it under the terms of the GNU Affero General Public License as published by
-the Free Software Foundation, either version 3 of the License, or
-(at your option) any later version.
-
-This program is distributed in the hope that it will be useful
-but WITHOUT ANY WARRANTY; without even the implied warranty of
-MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
-GNU Affero General Public License for more details.
-
-You should have received a copy of the GNU Affero General Public License
-along with this program. If not, see .
-*/
-
-package dataprotection
-
-import (
- "embed"
- "runtime"
- "time"
-
- corev1 "k8s.io/api/core/v1"
- metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
-
- viper "github.com/apecloud/kubeblocks/internal/viperx"
-)
-
-const (
- trueVal = "true"
-)
-
-const (
- // name of our custom finalizer
- dataProtectionFinalizerName = "dataprotection.kubeblocks.io/finalizer"
- // settings keys
- maxConcurDataProtectionReconKey = "MAXCONCURRENTRECONCILES_DATAPROTECTION"
-
- // label keys
- dataProtectionLabelBackupPolicyKey = "dataprotection.kubeblocks.io/backup-policy"
- dataProtectionLabelBackupTypeKey = "dataprotection.kubeblocks.io/backup-type"
- dataProtectionLabelAutoBackupKey = "dataprotection.kubeblocks.io/autobackup"
- dataProtectionLabelRestoreJobNameKey = "restorejobs.dataprotection.kubeblocks.io/name"
-
- dataProtectionBackupTargetPodKey = "dataprotection.kubeblocks.io/target-pod-name"
- dataProtectionAnnotationCreateByPolicyKey = "dataprotection.kubeblocks.io/created-by-policy"
-
- dataProtectionBackupRepoKey = "dataprotection.kubeblocks.io/backup-repo-name"
- dataProtectionNeedRepoPVCKey = "dataprotection.kubeblocks.io/need-repo-pvc"
-
- // annotation keys
- dataProtectionSecretTemplateMD5AnnotationKey = "dataprotection.kubeblocks.io/secret-template-md5"
- dataProtectionTemplateValuesMD5AnnotationKey = "dataprotection.kubeblocks.io/template-values-md5"
-
- // the key of persistentVolumeTemplate in the configmap.
- persistentVolumeTemplateKey = "persistentVolume"
-
- hostNameLabelKey = "kubernetes.io/hostname"
-)
-
-// condition constants
-const (
- // condition types
- ConditionTypeStorageProviderReady = "StorageProviderReady"
- ConditionTypeStorageClassCreated = "StorageClassCreated"
- ConditionTypeDerivedObjectsDeleted = "DerivedObjectsDeleted"
-
- // condition reasons
- ReasonStorageProviderReady = "StorageProviderReady"
- ReasonStorageProviderNotReady = "StorageProviderNotReady"
- ReasonStorageProviderNotFound = "StorageProviderNotFound"
- ReasonBadSecretTemplate = "BadSecretTemplate"
- ReasonBadStorageClassTemplate = "BadStorageClassTemplate"
- ReasonStorageClassCreated = "StorageClassCreated"
- ReasonHaveAssociatedBackups = "HaveAssociatedBackups"
- ReasonHaveResidualPVCs = "HaveResidualPVCs"
- ReasonDerivedObjectsDeleted = "DerivedObjectsDeleted"
- ReasonUnknownError = "UnknownError"
-)
-
-const manifestsUpdaterContainerName = "manifests-updater"
-
-var reconcileInterval = time.Second
-
-func init() {
- viper.SetDefault(maxConcurDataProtectionReconKey, runtime.NumCPU()*2)
-}
-
-var (
- //go:embed cue/*
- cueTemplates embed.FS
-)
-
-type backupPolicyOptions struct {
- Name string `json:"name"`
- BackupPolicyName string `json:"backupPolicyName"`
- Namespace string `json:"namespace"`
- MgrNamespace string `json:"mgrNamespace"`
- Cluster string `json:"cluster"`
- Schedule string `json:"schedule"`
- BackupType string `json:"backupType"`
- TTL metav1.Duration `json:"ttl,omitempty"`
- ServiceAccount string `json:"serviceAccount"`
- Image string `json:"image"`
- Tolerations *corev1.PodSpec `json:"tolerations"`
-}
diff --git a/controllers/dataprotection/types.go b/controllers/dataprotection/types.go
new file mode 100644
index 00000000000..883afd289b3
--- /dev/null
+++ b/controllers/dataprotection/types.go
@@ -0,0 +1,92 @@
+/*
+Copyright (C) 2022-2023 ApeCloud Co., Ltd
+
+This file is part of KubeBlocks project
+
+This program is free software: you can redistribute it and/or modify
+it under the terms of the GNU Affero General Public License as published by
+the Free Software Foundation, either version 3 of the License, or
+(at your option) any later version.
+
+This program is distributed in the hope that it will be useful
+but WITHOUT ANY WARRANTY; without even the implied warranty of
+MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+GNU Affero General Public License for more details.
+
+You should have received a copy of the GNU Affero General Public License
+along with this program. If not, see .
+*/
+
+package dataprotection
+
+import (
+ "runtime"
+ "time"
+
+ viper "github.com/apecloud/kubeblocks/internal/viperx"
+)
+
+const (
+ trueVal = "true"
+)
+
+const (
+ // settings keys
+ maxConcurDataProtectionReconKey = "MAXCONCURRENTRECONCILES_DATAPROTECTION"
+
+ // label keys
+ dataProtectionLabelBackupScheduleKey = "dataprotection.kubeblocks.io/backup-schedule"
+ dataProtectionLabelBackupPolicyKey = "dataprotection.kubeblocks.io/backup-policy"
+ dataProtectionLabelBackupMethodKey = "dataprotection.kubeblocks.io/backup-method"
+ dataProtectionLabelBackupTypeKey = "dataprotection.kubeblocks.io/backup-type"
+ dataProtectionLabelAutoBackupKey = "dataprotection.kubeblocks.io/autobackup"
+
+ dataProtectionBackupTargetPodKey = "dataprotection.kubeblocks.io/target-pod-name"
+ dataProtectionAnnotationCreateByPolicyKey = "dataprotection.kubeblocks.io/created-by-policy"
+
+ dataProtectionBackupRepoKey = "dataprotection.kubeblocks.io/backup-repo-name"
+ dataProtectionWaitRepoPreparationKey = "dataprotection.kubeblocks.io/wait-repo-preparation"
+ dataProtectionIsToolConfigKey = "dataprotection.kubeblocks.io/is-tool-config"
+
+ // annotation keys
+ dataProtectionSecretTemplateMD5AnnotationKey = "dataprotection.kubeblocks.io/secret-template-md5"
+ dataProtectionTemplateValuesMD5AnnotationKey = "dataprotection.kubeblocks.io/template-values-md5"
+ dataProtectionPVCTemplateMD5MD5AnnotationKey = "dataprotection.kubeblocks.io/pvc-template-md5"
+ dataProtectionToolConfigTemplateMD5MD5AnnotationKey = "dataprotection.kubeblocks.io/tool-config-template-md5"
+)
+
+// condition constants
+const (
+ // condition types
+ ConditionTypeStorageProviderReady = "StorageProviderReady"
+ ConditionTypeParametersChecked = "ParametersChecked"
+ ConditionTypeStorageClassCreated = "StorageClassCreated"
+ ConditionTypePVCTemplateChecked = "PVCTemplateChecked"
+ ConditionTypeToolConfigChecked = "ToolConfigSecretChecked"
+ ConditionTypeDerivedObjectsDeleted = "DerivedObjectsDeleted"
+
+ // condition reasons
+ ReasonStorageProviderReady = "StorageProviderReady"
+ ReasonStorageProviderNotReady = "StorageProviderNotReady"
+ ReasonStorageProviderNotFound = "StorageProviderNotFound"
+ ReasonInvalidStorageProvider = "InvalidStorageProvider"
+ ReasonParametersChecked = "ParametersChecked"
+ ReasonCredentialSecretNotFound = "CredentialSecretNotFound"
+ ReasonPrepareCSISecretFailed = "PrepareCSISecretFailed"
+ ReasonPrepareStorageClassFailed = "PrepareStorageClassFailed"
+ ReasonBadPVCTemplate = "BadPVCTemplate"
+ ReasonBadToolConfigTemplate = "BadToolConfigTemplate"
+ ReasonStorageClassCreated = "StorageClassCreated"
+ ReasonPVCTemplateChecked = "PVCTemplateChecked"
+ ReasonToolConfigChecked = "ToolConfigChecked"
+ ReasonHaveAssociatedBackups = "HaveAssociatedBackups"
+ ReasonHaveResidualPVCs = "HaveResidualPVCs"
+ ReasonDerivedObjectsDeleted = "DerivedObjectsDeleted"
+ ReasonUnknownError = "UnknownError"
+)
+
+var reconcileInterval = time.Second
+
+func init() {
+ viper.SetDefault(maxConcurDataProtectionReconKey, runtime.NumCPU()*2)
+}
diff --git a/controllers/dataprotection/utils.go b/controllers/dataprotection/utils.go
index 7ac4d23d219..2673180976a 100644
--- a/controllers/dataprotection/utils.go
+++ b/controllers/dataprotection/utils.go
@@ -22,180 +22,148 @@ package dataprotection
import (
"context"
"fmt"
- "strconv"
+ "sort"
"strings"
"sync"
- snapshotv1 "github.com/kubernetes-csi/external-snapshotter/client/v6/apis/volumesnapshot/v1"
- batchv1 "k8s.io/api/batch/v1"
corev1 "k8s.io/api/core/v1"
+ metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
"k8s.io/apimachinery/pkg/types"
- "k8s.io/apimachinery/pkg/util/json"
"k8s.io/client-go/tools/record"
ctrl "sigs.k8s.io/controller-runtime"
"sigs.k8s.io/controller-runtime/pkg/client"
- dataprotectionv1alpha1 "github.com/apecloud/kubeblocks/apis/dataprotection/v1alpha1"
+ appsv1alpha1 "github.com/apecloud/kubeblocks/apis/apps/v1alpha1"
+ dpv1alpha1 "github.com/apecloud/kubeblocks/apis/dataprotection/v1alpha1"
"github.com/apecloud/kubeblocks/internal/constant"
intctrlutil "github.com/apecloud/kubeblocks/internal/controllerutil"
- viper "github.com/apecloud/kubeblocks/internal/viperx"
+ dptypes "github.com/apecloud/kubeblocks/internal/dataprotection/types"
)
var (
errNoDefaultBackupRepo = fmt.Errorf("no default BackupRepo found")
)
-// byBackupStartTime sorts a list of jobs by start timestamp, using their names as a tie breaker.
-type byBackupStartTime []dataprotectionv1alpha1.Backup
-
-// Len returns the length of byBackupStartTime, for the sort.Sort
-func (o byBackupStartTime) Len() int { return len(o) }
-
-// Swap the items, for the sort.Sort
-func (o byBackupStartTime) Swap(i, j int) { o[i], o[j] = o[j], o[i] }
-
-// Less defines how to compare items, for the sort.Sort
-func (o byBackupStartTime) Less(i, j int) bool {
- if o[i].Status.StartTimestamp == nil && o[j].Status.StartTimestamp != nil {
- return false
+func getBackupPolicyByName(
+ reqCtx intctrlutil.RequestCtx,
+ cli client.Client,
+ name string) (*dpv1alpha1.BackupPolicy, error) {
+ backupPolicy := &dpv1alpha1.BackupPolicy{}
+ key := client.ObjectKey{
+ Namespace: reqCtx.Req.Namespace,
+ Name: name,
}
- if o[i].Status.StartTimestamp != nil && o[j].Status.StartTimestamp == nil {
- return true
- }
- if o[i].Status.StartTimestamp.Equal(o[j].Status.StartTimestamp) {
- return o[i].Name < o[j].Name
+ if err := cli.Get(reqCtx.Ctx, key, backupPolicy); err != nil {
+ return nil, err
}
- return o[i].Status.StartTimestamp.Before(o[j].Status.StartTimestamp)
+ return backupPolicy, nil
}
-// getBackupToolByName gets the backupTool by name.
-func getBackupToolByName(reqCtx intctrlutil.RequestCtx, cli client.Client, backupName string) (*dataprotectionv1alpha1.BackupTool, error) {
- backupTool := &dataprotectionv1alpha1.BackupTool{}
- backupToolNameSpaceName := types.NamespacedName{
- Name: backupName,
+// getActionSetByName gets the ActionSet by name.
+func getActionSetByName(reqCtx intctrlutil.RequestCtx,
+ cli client.Client, name string) (*dpv1alpha1.ActionSet, error) {
+ if name == "" {
+ return nil, nil
}
- if err := cli.Get(reqCtx.Ctx, backupToolNameSpaceName, backupTool); err != nil {
- reqCtx.Log.Error(err, "Unable to get backupTool for backup.", "BackupTool", backupToolNameSpaceName)
+ as := &dpv1alpha1.ActionSet{}
+ if err := cli.Get(reqCtx.Ctx, client.ObjectKey{Name: name}, as); err != nil {
+ reqCtx.Log.Error(err, "failed to get ActionSet for backup.", "ActionSet", name)
return nil, err
}
- return backupTool, nil
+ return as, nil
}
-// getCreatedCRNameByBackupPolicy gets the CR name which is created by BackupPolicy, such as CronJob/logfile Backup.
-func getCreatedCRNameByBackupPolicy(backupPolicy *dataprotectionv1alpha1.BackupPolicy, backupType dataprotectionv1alpha1.BackupType) string {
- name := fmt.Sprintf("%s-%s", generateUniqueNameWithBackupPolicy(backupPolicy), backupPolicy.Namespace)
- if len(name) > 30 {
- name = strings.TrimRight(name[:30], "-")
+func getBackupMethodByName(name string, backupPolicy *dpv1alpha1.BackupPolicy) *dpv1alpha1.BackupMethod {
+ for _, m := range backupPolicy.Spec.BackupMethods {
+ if m.Name == name {
+ return &m
+ }
}
- return fmt.Sprintf("%s-%s", name, string(backupType))
-}
-
-func getClusterLabelKeys() []string {
- return []string{constant.AppInstanceLabelKey, constant.KBAppComponentLabelKey}
-}
-
-func excludeLabelsForWorkload() []string {
- return []string{constant.KBAppComponentLabelKey}
+ return nil
}
-func buildAutoCreationAnnotations(backupPolicyName string) map[string]string {
- return map[string]string{
- dataProtectionAnnotationCreateByPolicyKey: "true",
- dataProtectionLabelBackupPolicyKey: backupPolicyName,
+// getTargetPods gets the target pods by BackupPolicy. If podName is not empty,
+// it will return the pod which name is podName. Otherwise, it will return the
+// pods which are selected by BackupPolicy selector and strategy.
+func getTargetPods(reqCtx intctrlutil.RequestCtx,
+ cli client.Client, podName string,
+ backupPolicy *dpv1alpha1.BackupPolicy) ([]*corev1.Pod, error) {
+ selector := backupPolicy.Spec.Target.PodSelector
+ if selector == nil || selector.LabelSelector == nil {
+ return nil, nil
}
-}
-// getBackupDestinationPath gets the destination path to storage backup datas.
-func getBackupDestinationPath(backup *dataprotectionv1alpha1.Backup, pathPrefix string) string {
- pathPrefix = strings.TrimRight(pathPrefix, "/")
- if strings.TrimSpace(pathPrefix) == "" || strings.HasPrefix(pathPrefix, "/") {
- return fmt.Sprintf("/%s%s/%s", backup.Namespace, pathPrefix, backup.Name)
+ labelSelector, err := metav1.LabelSelectorAsSelector(selector.LabelSelector)
+ if err != nil {
+ return nil, err
+ }
+ pods := &corev1.PodList{}
+ if err = cli.List(reqCtx.Ctx, pods,
+ client.InNamespace(reqCtx.Req.Namespace),
+ client.MatchingLabelsSelector{Selector: labelSelector}); err != nil {
+ return nil, err
}
- return fmt.Sprintf("/%s/%s/%s", backup.Namespace, pathPrefix, backup.Name)
-}
-// buildBackupWorkloadsLabels builds the labels for workloads which owned by backup.
-func buildBackupWorkloadsLabels(backup *dataprotectionv1alpha1.Backup) map[string]string {
- labels := backup.Labels
- if labels == nil {
- labels = map[string]string{}
- } else {
- for _, v := range excludeLabelsForWorkload() {
- delete(labels, v)
- }
+ if len(pods.Items) == 0 {
+ return nil, fmt.Errorf("failed to find target pods by backup policy %s/%s",
+ backupPolicy.Namespace, backupPolicy.Name)
}
- labels[constant.DataProtectionLabelBackupNameKey] = backup.Name
- return labels
-}
-func addTolerations(podSpec *corev1.PodSpec) (err error) {
- if cmTolerations := viper.GetString(constant.CfgKeyCtrlrMgrTolerations); cmTolerations != "" {
- if err = json.Unmarshal([]byte(cmTolerations), &podSpec.Tolerations); err != nil {
- return err
+ var targetPods []*corev1.Pod
+ if podName != "" {
+ for _, pod := range pods.Items {
+ if pod.Name == podName {
+ targetPods = append(targetPods, &pod)
+ break
+ }
}
- }
- if cmAffinity := viper.GetString(constant.CfgKeyCtrlrMgrAffinity); cmAffinity != "" {
- if err = json.Unmarshal([]byte(cmAffinity), &podSpec.Affinity); err != nil {
- return err
+ if len(targetPods) > 0 {
+ return targetPods, nil
}
}
- if cmNodeSelector := viper.GetString(constant.CfgKeyCtrlrMgrNodeSelector); cmNodeSelector != "" {
- if err = json.Unmarshal([]byte(cmNodeSelector), &podSpec.NodeSelector); err != nil {
- return err
+
+ strategy := selector.Strategy
+ sort.Sort(intctrlutil.ByPodName(pods.Items))
+ // if pod selection strategy is Any, always return first pod
+ switch strategy {
+ case dpv1alpha1.PodSelectionStrategyAny:
+ if len(pods.Items) > 0 {
+ targetPods = append(targetPods, &pods.Items[0])
+ }
+ case dpv1alpha1.PodSelectionStrategyAll:
+ for i := range pods.Items {
+ targetPods = append(targetPods, &pods.Items[i])
}
}
- return nil
+
+ return targetPods, nil
}
-// getIntervalSecondsForLogfile gets the interval seconds for logfile schedule cronExpression.
-// currently, only the fields of minutes and hours are taken and contain expressions such as '*/'.
-// If there is no such field, the default return is 60s.
-func getIntervalSecondsForLogfile(backupType dataprotectionv1alpha1.BackupType, cronExpression string) string {
- if backupType != dataprotectionv1alpha1.BackupTypeLogFile {
- return ""
- }
- // move time zone field
- if strings.HasPrefix(cronExpression, "TZ=") || strings.HasPrefix(cronExpression, "CRON_TZ=") {
- i := strings.Index(cronExpression, " ")
- cronExpression = strings.TrimSpace(cronExpression[i:])
- }
- var interval = "60"
- // skip the macro syntax
- if strings.HasPrefix(cronExpression, "@") {
- return interval + "s"
- }
- fields := strings.Fields(cronExpression)
-loop:
- for i, v := range fields {
- switch i {
- case 0:
- if strings.HasPrefix(v, "*/") {
- m, _ := strconv.Atoi(strings.ReplaceAll(v, "*/", ""))
- interval = strconv.Itoa(m * 60)
- break loop
- }
- case 1:
- if strings.HasPrefix(v, "*/") {
- m, _ := strconv.Atoi(strings.ReplaceAll(v, "*/", ""))
- interval = strconv.Itoa(m * 60 * 60)
- break loop
- }
- default:
- break loop
- }
- }
- return interval + "s"
+// getCluster gets the cluster and will ignore the error.
+func getCluster(ctx context.Context,
+ cli client.Client,
+ targetPod *corev1.Pod) *appsv1alpha1.Cluster {
+ clusterName := targetPod.Labels[constant.AppInstanceLabelKey]
+ if len(clusterName) == 0 {
+ return nil
+ }
+ cluster := &appsv1alpha1.Cluster{}
+ if err := cli.Get(ctx, client.ObjectKey{
+ Namespace: targetPod.Namespace,
+ Name: clusterName,
+ }, cluster); err != nil {
+ // should not affect the backup status
+ return nil
+ }
+ return cluster
}
-// filterCreatedByPolicy filters the workloads which are create by backupPolicy.
-func filterCreatedByPolicy(object client.Object) bool {
- labels := object.GetLabels()
- _, containsPolicyNameLabel := labels[dataProtectionLabelBackupPolicyKey]
- return labels[dataProtectionLabelAutoBackupKey] == "true" && containsPolicyNameLabel
+func getClusterLabelKeys() []string {
+ return []string{constant.AppInstanceLabelKey, constant.KBAppComponentLabelKey}
}
// sendWarningEventForError sends warning event for backup controller error
-func sendWarningEventForError(recorder record.EventRecorder, backup *dataprotectionv1alpha1.Backup, err error) {
+func sendWarningEventForError(recorder record.EventRecorder, backup *dpv1alpha1.Backup, err error) {
controllerErr := intctrlutil.UnwrapControllerError(err)
if controllerErr != nil {
recorder.Eventf(backup, corev1.EventTypeWarning, string(controllerErr.Type), err.Error())
@@ -205,77 +173,17 @@ func sendWarningEventForError(recorder record.EventRecorder, backup *dataprotect
}
}
-var configVolumeSnapshotError = []string{
- "Failed to set default snapshot class with error",
- "Failed to get snapshot class with error",
- "Failed to create snapshot content with error cannot find CSI PersistentVolumeSource for volume",
-}
-
-func isVolumeSnapshotConfigError(snap *snapshotv1.VolumeSnapshot) bool {
- if snap.Status == nil || snap.Status.Error == nil || snap.Status.Error.Message == nil {
- return false
- }
- for _, errMsg := range configVolumeSnapshotError {
- if strings.Contains(*snap.Status.Error.Message, errMsg) {
- return true
- }
- }
- return false
-}
-
-func generateJSON(path string, value string) string {
- segments := strings.Split(path, ".")
- jsonString := value
- for i := len(segments) - 1; i >= 0; i-- {
- jsonString = fmt.Sprintf(`{\"%s\":%s}`, segments[i], jsonString)
- }
- return jsonString
-}
-
-// cropJobName job name cannot exceed 63 characters for label name limit.
-func cropJobName(jobName string) string {
- if len(jobName) > 63 {
- return jobName[:63]
- }
- return jobName
-}
-
-func buildBackupInfoENV(backupDestinationPath string) string {
- return backupPathBase + backupDestinationPath + "/backup.info"
-}
-
-func generateUniqueNameWithBackupPolicy(backupPolicy *dataprotectionv1alpha1.BackupPolicy) string {
- uniqueName := backupPolicy.Name
- if len(backupPolicy.OwnerReferences) > 0 {
- uniqueName = fmt.Sprintf("%s-%s", backupPolicy.OwnerReferences[0].UID[:8], backupPolicy.OwnerReferences[0].Name)
- }
- return uniqueName
-}
-
-func generateUniqueJobName(backup *dataprotectionv1alpha1.Backup, prefix string) string {
- return cropJobName(fmt.Sprintf("%s-%s-%s", prefix, backup.UID[:8], backup.Name))
-}
-
-func buildDeleteBackupFilesJobNamespacedName(backup *dataprotectionv1alpha1.Backup) types.NamespacedName {
- jobName := fmt.Sprintf("%s-%s%s", backup.UID[:8], deleteBackupFilesJobNamePrefix, backup.Name)
- if len(jobName) > 63 {
- jobName = jobName[:63]
- }
- return types.NamespacedName{Namespace: backup.Namespace, Name: jobName}
-}
-
-func getDefaultBackupRepo(ctx context.Context, cli client.Client) (*dataprotectionv1alpha1.BackupRepo, error) {
- backupRepoList := &dataprotectionv1alpha1.BackupRepoList{}
- err := cli.List(ctx, backupRepoList)
- if err != nil {
+func getDefaultBackupRepo(ctx context.Context, cli client.Client) (*dpv1alpha1.BackupRepo, error) {
+ backupRepoList := &dpv1alpha1.BackupRepoList{}
+ if err := cli.List(ctx, backupRepoList); err != nil {
return nil, err
}
- var defaultRepo *dataprotectionv1alpha1.BackupRepo
+ var defaultRepo *dpv1alpha1.BackupRepo
for idx := range backupRepoList.Items {
repo := &backupRepoList.Items[idx]
// skip non-default repo
- if !(repo.Annotations[constant.DefaultBackupRepoAnnotationKey] == trueVal &&
- repo.Status.Phase == dataprotectionv1alpha1.BackupRepoReady) {
+ if !(repo.Annotations[dptypes.DefaultBackupRepoAnnotationKey] == trueVal &&
+ repo.Status.Phase == dpv1alpha1.BackupRepoReady) {
continue
}
if defaultRepo != nil {
@@ -391,12 +299,3 @@ func fromFlattenName(flatten string) (name string, namespace string) {
}
return
}
-
-func containsJobCondition(job *batchv1.Job, jobCondType batchv1.JobConditionType) bool {
- for _, jobCond := range job.Status.Conditions {
- if jobCond.Type == jobCondType {
- return true
- }
- }
- return false
-}
diff --git a/controllers/extensions/addon_controller.go b/controllers/extensions/addon_controller.go
index 496f2660b0a..8622bf68663 100644
--- a/controllers/extensions/addon_controller.go
+++ b/controllers/extensions/addon_controller.go
@@ -36,7 +36,6 @@ import (
"sigs.k8s.io/controller-runtime/pkg/handler"
"sigs.k8s.io/controller-runtime/pkg/log"
"sigs.k8s.io/controller-runtime/pkg/reconcile"
- "sigs.k8s.io/controller-runtime/pkg/source"
extensionsv1alpha1 "github.com/apecloud/kubeblocks/apis/extensions/v1alpha1"
"github.com/apecloud/kubeblocks/internal/constant"
@@ -147,14 +146,14 @@ func (r *AddonReconciler) Reconcile(ctx context.Context, req ctrl.Request) (ctrl
func (r *AddonReconciler) SetupWithManager(mgr ctrl.Manager) error {
return ctrl.NewControllerManagedBy(mgr).
For(&extensionsv1alpha1.Addon{}).
- Watches(&source.Kind{Type: &batchv1.Job{}}, handler.EnqueueRequestsFromMapFunc(r.findAddonJobs)).
+ Watches(&batchv1.Job{}, handler.EnqueueRequestsFromMapFunc(r.findAddonJobs)).
WithOptions(controller.Options{
MaxConcurrentReconciles: viper.GetInt(maxConcurrentReconcilesKey),
}).
Complete(r)
}
-func (r *AddonReconciler) findAddonJobs(job client.Object) []reconcile.Request {
+func (r *AddonReconciler) findAddonJobs(ctx context.Context, job client.Object) []reconcile.Request {
labels := job.GetLabels()
if _, ok := labels[constant.AddonNameLabelKey]; !ok {
return []reconcile.Request{}
diff --git a/controllers/extensions/addon_controller_test.go b/controllers/extensions/addon_controller_test.go
index 205453fce0b..361b3dfcd5d 100644
--- a/controllers/extensions/addon_controller_test.go
+++ b/controllers/extensions/addon_controller_test.go
@@ -40,7 +40,7 @@ import (
extensionsv1alpha1 "github.com/apecloud/kubeblocks/apis/extensions/v1alpha1"
"github.com/apecloud/kubeblocks/internal/constant"
- intctrlutil "github.com/apecloud/kubeblocks/internal/generics"
+ "github.com/apecloud/kubeblocks/internal/generics"
"github.com/apecloud/kubeblocks/internal/testutil"
testapps "github.com/apecloud/kubeblocks/internal/testutil/apps"
viper "github.com/apecloud/kubeblocks/internal/viperx"
@@ -55,21 +55,21 @@ var _ = Describe("Addon controller", func() {
By("clean resources")
// non-namespaced
ml := client.HasLabels{testCtx.TestObjLabelKey}
- testapps.ClearResourcesWithRemoveFinalizerOption(&testCtx, intctrlutil.AddonSignature, true, ml)
+ testapps.ClearResourcesWithRemoveFinalizerOption(&testCtx, generics.AddonSignature, true, ml)
inNS := client.InNamespace(viper.GetString(constant.CfgKeyCtrlrMgrNS))
- testapps.ClearResourcesWithRemoveFinalizerOption(&testCtx, intctrlutil.JobSignature, true, inNS,
+ testapps.ClearResourcesWithRemoveFinalizerOption(&testCtx, generics.JobSignature, true, inNS,
client.HasLabels{
constant.AddonNameLabelKey,
})
- testapps.ClearResourcesWithRemoveFinalizerOption(&testCtx, intctrlutil.JobSignature, true, inNS,
+ testapps.ClearResourcesWithRemoveFinalizerOption(&testCtx, generics.JobSignature, true, inNS,
client.HasLabels{
constant.AppManagedByLabelKey,
})
// delete rest mocked objects
- testapps.ClearResources(&testCtx, intctrlutil.ConfigMapSignature, inNS, ml)
- testapps.ClearResources(&testCtx, intctrlutil.SecretSignature, inNS, ml)
+ testapps.ClearResources(&testCtx, generics.ConfigMapSignature, inNS, ml)
+ testapps.ClearResources(&testCtx, generics.SecretSignature, inNS, ml)
// By("deleting the Namespace to perform the tests")
// Eventually(func(g Gomega) {
diff --git a/controllers/k8score/const.go b/controllers/k8score/const.go
index d47432f46de..c07ce34506a 100644
--- a/controllers/k8score/const.go
+++ b/controllers/k8score/const.go
@@ -22,15 +22,4 @@ package k8score
const (
// roleChangedAnnotKey is used to mark the role change event has been handled.
roleChangedAnnotKey = "role.kubeblocks.io/event-handled"
-
- // TrueStr values
- trueStr = "true"
-)
-
-const (
- ProbeEventOperationNotImpl ProbeEventType = "OperationNotImplemented"
- ProbeEventCheckRoleFailed ProbeEventType = "Failed"
- ProbeEventRoleInvalid ProbeEventType = "roleInvalid"
- ProbeEventRoleChanged ProbeEventType = "roleChanged"
- ProbeEventRoleUnChanged ProbeEventType = "roleUnChanged"
)
diff --git a/controllers/k8score/event_controller_test.go b/controllers/k8score/event_controller_test.go
index 53215fe656f..f461d0714e2 100644
--- a/controllers/k8score/event_controller_test.go
+++ b/controllers/k8score/event_controller_test.go
@@ -35,6 +35,7 @@ import (
"sigs.k8s.io/controller-runtime/pkg/log"
appsv1alpha1 "github.com/apecloud/kubeblocks/apis/apps/v1alpha1"
+ workloads "github.com/apecloud/kubeblocks/apis/workloads/v1alpha1"
"github.com/apecloud/kubeblocks/internal/constant"
"github.com/apecloud/kubeblocks/internal/controller/builder"
intctrlutil "github.com/apecloud/kubeblocks/internal/controllerutil"
@@ -123,9 +124,30 @@ var _ = Describe("Event Controller", func() {
Create(&testCtx).GetObject()
Eventually(testapps.CheckObjExists(&testCtx, client.ObjectKeyFromObject(clusterObj), &appsv1alpha1.Cluster{}, true)).Should(Succeed())
+ rsmName := fmt.Sprintf("%s-%s", clusterObj.Name, consensusCompName)
+ rsm := testapps.NewRSMFactory(clusterObj.Namespace, rsmName, clusterObj.Name, consensusCompName).
+ SetReplicas(int32(3)).
+ AddContainer(corev1.Container{Name: testapps.DefaultMySQLContainerName, Image: testapps.ApeCloudMySQLImage}).
+ Create(&testCtx).GetObject()
+ Expect(testapps.GetAndChangeObj(&testCtx, client.ObjectKeyFromObject(rsm), func(tmpRSM *workloads.ReplicatedStateMachine) {
+ tmpRSM.Spec.Roles = []workloads.ReplicaRole{
+ {
+ Name: "leader",
+ IsLeader: true,
+ AccessMode: workloads.ReadWriteMode,
+ CanVote: true,
+ },
+ {
+ Name: "follower",
+ IsLeader: false,
+ AccessMode: workloads.ReadonlyMode,
+ CanVote: true,
+ },
+ }
+ })()).Should(Succeed())
By("create involved pod")
var uid types.UID
- podName := "foo"
+ podName := fmt.Sprintf("%s-%d", rsmName, 0)
pod := createInvolvedPod(podName, clusterObj.Name, consensusCompName)
Expect(testCtx.CreateObj(ctx, pod)).Should(Succeed())
Eventually(func() error {
@@ -159,13 +181,13 @@ var _ = Describe("Event Controller", func() {
g.Expect(p).ShouldNot(BeNil())
g.Expect(p.Labels).ShouldNot(BeNil())
g.Expect(p.Labels[constant.RoleLabelKey]).Should(Equal(role))
- g.Expect(p.Annotations[constant.LastRoleChangedEventTimestampAnnotationKey]).Should(Equal(sndEvent.EventTime.Time.Format(time.RFC3339Nano)))
+ g.Expect(p.Annotations[constant.LastRoleSnapshotVersionAnnotationKey]).Should(Equal(sndEvent.EventTime.Time.Format(time.RFC3339Nano)))
})).Should(Succeed())
Eventually(testapps.CheckObj(&testCtx, client.ObjectKeyFromObject(sndEvent), func(g Gomega, e *corev1.Event) {
g.Expect(e).ShouldNot(BeNil())
g.Expect(e.Annotations).ShouldNot(BeNil())
- g.Expect(e.Annotations[roleChangedAnnotKey]).Should(Equal(trueStr))
+ g.Expect(e.Annotations[roleChangedAnnotKey]).Should(Equal("count-0"))
})).Should(Succeed())
By("check whether the duration and number of events reach the threshold")
@@ -190,7 +212,7 @@ var _ = Describe("Event Controller", func() {
g.Expect(p).ShouldNot(BeNil())
g.Expect(p.Labels).ShouldNot(BeNil())
g.Expect(p.Labels[constant.RoleLabelKey]).ShouldNot(Equal(role))
- g.Expect(p.Annotations[constant.LastRoleChangedEventTimestampAnnotationKey]).ShouldNot(Equal(sndInvalidEvent.EventTime.Time.Format(time.RFC3339Nano)))
+ g.Expect(p.Annotations[constant.LastRoleSnapshotVersionAnnotationKey]).ShouldNot(Equal(sndInvalidEvent.EventTime.Time.Format(time.RFC3339Nano)))
})).Should(Succeed())
By("send role changed event with afterLastTS later than pod last role changes event timestamp annotation should be update successfully")
@@ -212,7 +234,7 @@ var _ = Describe("Event Controller", func() {
g.Expect(p).ShouldNot(BeNil())
g.Expect(p.Labels).ShouldNot(BeNil())
g.Expect(p.Labels[constant.RoleLabelKey]).Should(Equal(role))
- g.Expect(p.Annotations[constant.LastRoleChangedEventTimestampAnnotationKey]).Should(Equal(sndValidEvent.EventTime.Time.Format(time.RFC3339Nano)))
+ g.Expect(p.Annotations[constant.LastRoleSnapshotVersionAnnotationKey]).Should(Equal(sndValidEvent.EventTime.Time.Format(time.RFC3339Nano)))
})).Should(Succeed())
})
})
diff --git a/controllers/k8score/event_handler.go b/controllers/k8score/event_handler.go
index 21ec2b1cbea..7e002904183 100644
--- a/controllers/k8score/event_handler.go
+++ b/controllers/k8score/event_handler.go
@@ -35,6 +35,5 @@ type EventHandler interface {
var EventHandlerMap = map[string]EventHandler{}
func init() {
- EventHandlerMap["role-change-handler"] = &RoleChangeEventHandler{}
EventHandlerMap["rsm-event-handler"] = &rsm.PodRoleEventHandler{}
}
diff --git a/controllers/k8score/role_change_event_handler.go b/controllers/k8score/role_change_event_handler.go
deleted file mode 100644
index 6c4d2e63b43..00000000000
--- a/controllers/k8score/role_change_event_handler.go
+++ /dev/null
@@ -1,224 +0,0 @@
-/*
-Copyright (C) 2022-2023 ApeCloud Co., Ltd
-
-This file is part of KubeBlocks project
-
-This program is free software: you can redistribute it and/or modify
-it under the terms of the GNU Affero General Public License as published by
-the Free Software Foundation, either version 3 of the License, or
-(at your option) any later version.
-
-This program is distributed in the hope that it will be useful
-but WITHOUT ANY WARRANTY; without even the implied warranty of
-MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
-GNU Affero General Public License for more details.
-
-You should have received a copy of the GNU Affero General Public License
-along with this program. If not, see .
-*/
-
-package k8score
-
-import (
- "strings"
- "time"
-
- corev1 "k8s.io/api/core/v1"
- "k8s.io/apimachinery/pkg/types"
- "k8s.io/client-go/tools/record"
- "sigs.k8s.io/controller-runtime/pkg/client"
-
- appsv1alpha1 "github.com/apecloud/kubeblocks/apis/apps/v1alpha1"
- "github.com/apecloud/kubeblocks/controllers/apps/components"
- "github.com/apecloud/kubeblocks/internal/constant"
- intctrlutil "github.com/apecloud/kubeblocks/internal/controllerutil"
- lorryutil "github.com/apecloud/kubeblocks/lorry/util"
-)
-
-// RoleChangeEventHandler is the event handler for the role change event
-type RoleChangeEventHandler struct{}
-
-var _ EventHandler = &RoleChangeEventHandler{}
-
-// var term int
-
-// Handle handles role changed event.
-func (r *RoleChangeEventHandler) Handle(cli client.Client, reqCtx intctrlutil.RequestCtx, recorder record.EventRecorder, event *corev1.Event) error {
- if event.Reason != string(lorryutil.CheckRoleOperation) {
- return nil
- }
- var (
- err error
- annotations = event.GetAnnotations()
- )
- // filter role changed event that has been handled
- if annotations != nil && annotations[roleChangedAnnotKey] == trueStr {
- return nil
- }
-
- if _, err = handleRoleChangedEvent(cli, reqCtx, recorder, event); err != nil {
- return err
- }
-
- // event order is crucial in role probing, but it's not guaranteed when controller restarted, so we have to mark them to be filtered
- patch := client.MergeFrom(event.DeepCopy())
- if event.Annotations == nil {
- event.Annotations = make(map[string]string, 0)
- }
- event.Annotations[roleChangedAnnotKey] = trueStr
- return cli.Patch(reqCtx.Ctx, event, patch)
-}
-
-// handleRoleChangedEvent handles role changed event and return role.
-func handleRoleChangedEvent(cli client.Client, reqCtx intctrlutil.RequestCtx, recorder record.EventRecorder, event *corev1.Event) (string, error) {
- // parse probe event message
- message := ParseProbeEventMessage(reqCtx, event)
- if message == nil {
- reqCtx.Log.Info("parse probe event message failed", "message", event.Message)
- return "", nil
- }
-
- // if probe event operation is not implemented, check role failed or invalid, ignore it
- if message.Event == ProbeEventOperationNotImpl || message.Event == ProbeEventCheckRoleFailed || message.Event == ProbeEventRoleInvalid {
- reqCtx.Log.Info("probe event failed", "message", message.Message)
- return "", nil
- }
- role := strings.ToLower(message.Role)
-
- podName := types.NamespacedName{
- Namespace: event.InvolvedObject.Namespace,
- Name: event.InvolvedObject.Name,
- }
- // get pod
- pod := &corev1.Pod{}
- if err := cli.Get(reqCtx.Ctx, podName, pod); err != nil {
- return role, err
- }
- // event belongs to old pod with the same name, ignore it
- if pod.UID != event.InvolvedObject.UID {
- return role, nil
- }
-
- // compare the EventTime of the current event object with the lastTimestamp of the last recorded in the pod annotation,
- // if the current event's EventTime is earlier than the recorded lastTimestamp in the pod annotation,
- // it indicates that the current event has arrived out of order and is expired, so it should not be processed.
- lastTimestampStr, ok := pod.Annotations[constant.LastRoleChangedEventTimestampAnnotationKey]
- if ok {
- lastTimestamp, err := time.Parse(time.RFC3339Nano, lastTimestampStr)
- if err != nil {
- reqCtx.Log.Info("failed to parse last role changed event timestamp from pod annotation", "pod", pod.Name, "error", err.Error())
- return role, err
- }
- eventLastTS := event.EventTime.Time
- if !eventLastTS.After(lastTimestamp) {
- reqCtx.Log.Info("event's EventTime is earlier than the recorded lastTimestamp in the pod annotation, it should not be processed.", "event uid", event.UID, "pod", pod.Name, "role", role, "originalRole", message.OriginalRole, "event EventTime", event.EventTime.Time.String(), "annotation lastTimestamp", lastTimestampStr)
- return role, nil
- }
- }
-
- // get cluster obj of the pod
- cluster := &appsv1alpha1.Cluster{}
- if err := cli.Get(reqCtx.Ctx, types.NamespacedName{
- Namespace: pod.Namespace,
- Name: pod.Labels[constant.AppInstanceLabelKey],
- }, cluster); err != nil {
- return role, err
- }
- reqCtx.Log.V(1).Info("handle role changed event", "event uid", event.UID, "cluster", cluster.Name, "pod", pod.Name, "role", role, "originalRole", message.OriginalRole)
- compName, componentDef, err := components.GetComponentInfoByPod(reqCtx.Ctx, cli, *cluster, pod)
- if err != nil {
- return role, err
- }
- switch componentDef.WorkloadType {
- case appsv1alpha1.Consensus:
- return role, components.UpdateConsensusSetRoleLabel(cli, reqCtx, event, componentDef, pod, role)
- case appsv1alpha1.Replication:
- return role, components.HandleReplicationSetRoleChangeEvent(cli, reqCtx, event, cluster, compName, pod, role)
- }
- return role, nil
-}
-
-// handleGlobalInfoEvent handles cluster role changed event and return err if occurs.
-// func handleGlobalInfoEvent(cli client.Client, reqCtx intctrlutil.RequestCtx, recorder record.EventRecorder, event *corev1.Event) error {
-// // parse probe event message
-// global := ParseProbeEventMessage(reqCtx, event)
-// if global == nil {
-// reqCtx.Log.Info("parse probe event message failed", "message", event.Message)
-// return nil
-// }
-//
-// // if probe event operation is not implemented, check role failed or invalid, ignore it
-// if global.Event == ProbeEventOperationNotImpl || global.Event == ProbeEventCheckRoleFailed || global.Event == ProbeEventRoleInvalid {
-// reqCtx.Log.Info("probe event failed")
-// return nil
-// }
-//
-// // check term
-// if global.Term < term {
-// reqCtx.Log.Info("out of date message", "message", event.Message)
-// return nil
-// }
-// term = global.Term
-//
-// // get pod
-// pod := &corev1.Pod{}
-// pods := &corev1.PodList{}
-// err := cli.List(reqCtx.Ctx, pods, client.InNamespace(event.InvolvedObject.Namespace))
-// if err != nil || len(pods.Items) == 0 {
-// return err
-// }
-// for _, p := range pods.Items {
-// if p.Name == event.InvolvedObject.Name {
-// pod = &p
-// break
-// }
-// }
-// // event belongs to old pod with the same name, ignore it
-// if pod.UID != event.InvolvedObject.UID {
-// return nil
-// }
-//
-// // get cluster obj of the pod
-// cluster := &appsv1alpha1.Cluster{}
-// if err := cli.Get(reqCtx.Ctx, types.NamespacedName{
-// Namespace: event.InvolvedObject.Namespace,
-// Name: pod.Labels[constant.AppInstanceLabelKey],
-// }, cluster); err != nil {
-// return err
-// }
-//
-// // get component name
-// reqCtx.Log.V(1).Info("handle role changed event", "event uid", event.UID, "cluster", cluster.Name, "pod", pod.Name)
-// compName, componentDef, err := components.GetComponentInfoByPod(reqCtx.Ctx, cli, *cluster, pod)
-// if err != nil {
-// return err
-// }
-//
-// // pod involved is a follower, just update single pod
-// if global.Message != "" {
-// role := strings.ToLower(global.Message)
-// return components.UpdateConsensusSetRoleLabel(cli, reqCtx, event, componentDef, pod, role)
-// }
-//
-// switch componentDef.WorkloadType {
-// case appsv1alpha1.Consensus:
-// for _, pod := range pods.Items {
-// if role, ok := global.PodName2Role[pod.Name]; ok {
-// err := components.UpdateConsensusSetRoleLabel(cli, reqCtx, event, componentDef, &pod, role)
-// if err != nil {
-// return err
-// }
-// }
-// }
-// case appsv1alpha1.Replication:
-// for _, pod := range pods.Items {
-// if role, ok := global.PodName2Role[pod.Status.PodIP]; ok {
-// err := components.HandleReplicationSetRoleChangeEvent(cli, reqCtx, event, cluster, compName, &pod, role)
-// if err != nil {
-// return err
-// }
-// }
-// }
-// }
-// return nil
-// }
diff --git a/controllers/k8score/suite_test.go b/controllers/k8score/suite_test.go
index 1557b7a067d..f81b2aea10a 100644
--- a/controllers/k8score/suite_test.go
+++ b/controllers/k8score/suite_test.go
@@ -40,6 +40,7 @@ import (
"sigs.k8s.io/controller-runtime/pkg/manager"
appsv1alpha1 "github.com/apecloud/kubeblocks/apis/apps/v1alpha1"
+ workloads "github.com/apecloud/kubeblocks/apis/workloads/v1alpha1"
"github.com/apecloud/kubeblocks/internal/testutil"
viper "github.com/apecloud/kubeblocks/internal/viperx"
)
@@ -95,6 +96,9 @@ var _ = BeforeSuite(func() {
err = appsv1alpha1.AddToScheme(scheme.Scheme)
Expect(err).NotTo(HaveOccurred())
+ err = workloads.AddToScheme(scheme.Scheme)
+ Expect(err).NotTo(HaveOccurred())
+
// +kubebuilder:scaffold:scheme
k8sClient, err = client.New(cfg, client.Options{Scheme: scheme.Scheme})
diff --git a/controllers/storage/storageprovider_controller.go b/controllers/storage/storageprovider_controller.go
index 3109358c152..8727be241a7 100644
--- a/controllers/storage/storageprovider_controller.go
+++ b/controllers/storage/storageprovider_controller.go
@@ -33,7 +33,6 @@ import (
"sigs.k8s.io/controller-runtime/pkg/handler"
"sigs.k8s.io/controller-runtime/pkg/log"
"sigs.k8s.io/controller-runtime/pkg/reconcile"
- "sigs.k8s.io/controller-runtime/pkg/source"
storagev1alpha1 "github.com/apecloud/kubeblocks/apis/storage/v1alpha1"
intctrlutil "github.com/apecloud/kubeblocks/internal/controllerutil"
@@ -194,8 +193,8 @@ func (r *StorageProviderReconciler) deleteExternalResources(
func (r *StorageProviderReconciler) SetupWithManager(mgr ctrl.Manager) error {
return ctrl.NewControllerManagedBy(mgr).
For(&storagev1alpha1.StorageProvider{}).
- Watches(&source.Kind{Type: &storagev1.CSIDriver{}},
- handler.EnqueueRequestsFromMapFunc(func(object client.Object) []reconcile.Request {
+ Watches(&storagev1.CSIDriver{},
+ handler.EnqueueRequestsFromMapFunc(func(ctx context.Context, object client.Object) []reconcile.Request {
r.mu.Lock()
defer r.mu.Unlock()
driverName := object.GetName()
diff --git a/controllers/workloads/replicatedstatemachine_controller.go b/controllers/workloads/replicatedstatemachine_controller.go
index 350484f4a47..47dee121b90 100644
--- a/controllers/workloads/replicatedstatemachine_controller.go
+++ b/controllers/workloads/replicatedstatemachine_controller.go
@@ -30,7 +30,6 @@ import (
ctrl "sigs.k8s.io/controller-runtime"
"sigs.k8s.io/controller-runtime/pkg/client"
"sigs.k8s.io/controller-runtime/pkg/log"
- "sigs.k8s.io/controller-runtime/pkg/source"
workloads "github.com/apecloud/kubeblocks/apis/workloads/v1alpha1"
"github.com/apecloud/kubeblocks/internal/constant"
@@ -151,9 +150,9 @@ func (r *ReplicatedStateMachineReconciler) SetupWithManager(mgr ctrl.Manager) er
return ctrl.NewControllerManagedBy(mgr).
For(&workloads.ReplicatedStateMachine{}).
- Watches(&source.Kind{Type: &appsv1.StatefulSet{}}, stsHandler).
- Watches(&source.Kind{Type: &batchv1.Job{}}, jobHandler).
- Watches(&source.Kind{Type: &corev1.Pod{}}, podHandler).
+ Watches(&appsv1.StatefulSet{}, stsHandler).
+ Watches(&batchv1.Job{}, jobHandler).
+ Watches(&corev1.Pod{}, podHandler).
Complete(r)
}
@@ -164,6 +163,6 @@ func (r *ReplicatedStateMachineReconciler) SetupWithManager(mgr ctrl.Manager) er
For(&workloads.ReplicatedStateMachine{}).
Owns(&appsv1.StatefulSet{}).
Owns(&batchv1.Job{}).
- Watches(&source.Kind{Type: &corev1.Pod{}}, podHandler).
+ Watches(&corev1.Pod{}, podHandler).
Complete(r)
}
diff --git a/deploy/apecloud-mysql/dataprotection/backup.sh b/deploy/apecloud-mysql/dataprotection/backup.sh
new file mode 100644
index 00000000000..507a214cbab
--- /dev/null
+++ b/deploy/apecloud-mysql/dataprotection/backup.sh
@@ -0,0 +1,12 @@
+#!/bin/bash
+set -e
+if [ -d ${DP_BACKUP_DIR} ]; then
+ rm -rf ${DP_BACKUP_DIR}
+fi
+mkdir -p ${DP_BACKUP_DIR}
+START_TIME=$(date -u '+%Y-%m-%dT%H:%M:%SZ')
+xtrabackup --compress=zstd --backup --safe-slave-backup --slave-info --stream=xbstream \
+ --host=${DP_DB_HOST} --user=${DP_DB_USER} --port=${DP_DB_PORT} --password=${DP_DB_PASSWORD} --datadir=${DATA_DIR} >${DP_BACKUP_DIR}/${DP_BACKUP_NAME}.xbstream
+STOP_TIME=$(date -u '+%Y-%m-%dT%H:%M:%SZ')
+TOTAL_SIZE=$(du -shx ${DP_BACKUP_DIR} | awk '{print $1}')
+echo "{\"totalSize\":\"$TOTAL_SIZE\",\"timeRange\":{\"start\":\"${START_TIME}\",\"end\":\"${STOP_TIME}\"}}" >${DP_BACKUP_DIR}/backup.info
diff --git a/deploy/apecloud-mysql/dataprotection/pitr-backup.sh b/deploy/apecloud-mysql/dataprotection/pitr-backup.sh
deleted file mode 100644
index c4694b4d0ca..00000000000
--- a/deploy/apecloud-mysql/dataprotection/pitr-backup.sh
+++ /dev/null
@@ -1,135 +0,0 @@
-#!/bin/bash
-
-# export wal-g environments
-backup_binlog_dir=${BACKUP_DIR}/${DP_TARGET_POD_NAME}
-export WALG_MYSQL_DATASOURCE_NAME="${DB_USER}:${DB_PASSWORD}@tcp(${DB_HOST}:${DP_DB_PORT})/mysql"
-export WALG_COMPRESSION_METHOD=zstd
-export WALG_FILE_PREFIX=${backup_binlog_dir}
-export WALG_MYSQL_CHECK_GTIDS=true
-export MYSQL_PWD=${DB_PASSWORD}
-
-# get binlog basename
-MYSQL_CMD="mysql -u ${DB_USER} -h ${DB_HOST} -N"
-log_bin_basename=$(${MYSQL_CMD} -e "SHOW VARIABLES LIKE 'log_bin_basename';" | awk -F'\t' '{print $2}')
-if [ -z ${log_bin_basename} ]; then
- echo "ERROR: pod/${DP_TARGET_POD_NAME} connect failed."
- exit 1
-fi
-LOG_DIR=$(dirname $log_bin_basename)
-LOG_PREFIX=$(basename $log_bin_basename)
-
-latest_bin_log=""
-last_flush_logs_time=$(date +%s)
-last_purge_time=$(date +%s)
-flush_bin_logs_interval=600
-
-if [[ ${FLUSH_BINLOG_INTERVAL_SECONDS} =~ ^[0-9]+$ ]];then
- flush_bin_logs_interval=${FLUSH_BINLOG_INTERVAL_SECONDS}
-fi
-
-function log() {
- msg=$1
- local curr_date=$(date -u '+%Y-%m-%d %H:%M:%S')
- echo "${curr_date} INFO: $msg"
-}
-
-# checks if the mysql process is ok
-function check_mysql_process() {
- is_ok=false
- for ((i=1;i<4;i++));do
- role=$(${MYSQL_CMD} -e "select role from information_schema.wesql_cluster_local;" | head -n 1)
- if [[ $? -eq 0 && (-z ${DP_TARGET_POD_ROLE} || "${DP_TARGET_POD_ROLE,,}" == "${role,,}") ]]; then
- is_ok=true
- break
- fi
- echo "Warning: target backup pod/${DP_TARGET_POD_NAME} is not OK, target role: ${DP_TARGET_POD_ROLE}, current role: ${role}, retry detection!"
- sleep 1
- done
- if [[ ${is_ok} == "false" ]];then
- echo "ERROR: target backup pod/${DP_TARGET_POD_NAME} is not OK, target role: ${DP_TARGET_POD_ROLE}, current role: ${role}!"
- exit 1
- fi
-}
-
-# clean up expired logfiles, interval is 60s
-function purge_expired_files() {
- local curr_time=$(date +%s)
- local diff_time=$((${curr_time}-${last_purge_time}))
- if [[ -z ${LOGFILE_TTL_SECOND} || ${diff_time} -lt 60 ]]; then
- return
- fi
- if [[ -d ${backup_binlog_dir}/binlog_005 ]];then
- local retention_minute=$((${LOGFILE_TTL_SECOND}/60))
- local fileCount=$(find ${backup_binlog_dir}/binlog_005 -mmin +${retention_minute} -name "*.zst" | wc -l)
- find ${backup_binlog_dir}/binlog_005 -mmin +${retention_minute} -name "*.zst" -exec rm -rf {} \;
- if [ ${fileCount} -gt 0 ]; then
- log "clean up expired binlog file successfully, file count: ${fileCount}"
- fi
- last_purge_time=${curr_time}
- fi
-}
-
-# flush bin logs, interval is 600s by default
-function flush_binlogs() {
- local curr_time=$(date +%s)
- local diff_time=$((${curr_time}-${last_flush_logs_time}))
- if [[ ${diff_time} -lt ${flush_bin_logs_interval} ]]; then
- return
- fi
- local LATEST_TRANS=$(mysqlbinlog $(ls -Ft $LOG_DIR/|grep -e '^mysql-bin.*[[:digit:]]$' |head -n 1)|grep 'Xid =' |head -n 1)
- # only flush bin logs when Xid exists
- if [[ -n "${LATEST_TRANS}" ]]; then
- log "flush binary logs"
- ${MYSQL_CMD} -e "flush binary logs";
- fi
- last_flush_logs_time=${curr_time}
-}
-
-# upload bin logs by wal-g
-function upload_bin_logs() {
- latest_bin_log=$(ls -Ftr $LOG_DIR/|grep -e "^${LOG_PREFIX}.*[[:digit:]]$"|tail -n 1)
- wal-g binlog-push;
-}
-
-function get_binlog_start_time() {
- local binlog=$1
- local time=$(mysqlbinlog ${binlog} | grep -m 1 "end_log_pos" | awk '{print $1, $2}'|tr -d '#')
- local time=$(date -d "$time" -u '+%Y-%m-%dT%H:%M:%SZ')
- echo $time
-}
-
-function save_backup_status() {
- local first_bin_log=$(ls -Ftr $LOG_DIR/|grep -e "^${LOG_PREFIX}.*[[:digit:]]$"|head -n 1)
- local START_TIME=$(get_binlog_start_time $first_bin_log)
- local STOP_TIME=$(get_binlog_start_time $latest_bin_log)
- local TOTAL_SIZE=$(du -shx ${BACKUP_DIR}|awk '{print $1}')
- if [[ -z $STOP_TIME ]];then
- echo "{\"totalSize\":\"$TOTAL_SIZE\",\"manifests\":{\"backupTool\":{\"uploadTotalSize\":\"${TOTAL_SIZE}\"}}}" > ${BACKUP_DIR}/backup.info
- else
- echo "{\"totalSize\":\"$TOTAL_SIZE\",\"manifests\":{\"backupLog\":{\"startTime\":\"${START_TIME}\",\"stopTime\":\"${STOP_TIME}\"},\"backupTool\":{\"uploadTotalSize\":\"${TOTAL_SIZE}\"}}}" > ${BACKUP_DIR}/backup.info
- fi
-}
-
-mkdir -p ${backup_binlog_dir} && cd $LOG_DIR
-# trap term signal
-trap "echo 'Terminating...' && sync && exit 0" TERM
-log "start to archive binlog logs"
-while true; do
-
- # check if mysql process is ok
- check_mysql_process
-
- # flush bin logs
- flush_binlogs
-
- # upload bin log
- upload_bin_logs
-
- # save backup status which will be updated to `backup` CR by the sidecar
- save_backup_status
-
- # purge the expired bin logs
- purge_expired_files
- sleep ${DP_INTERVAL_SECONDS}
-done
-
diff --git a/deploy/apecloud-mysql/dataprotection/pitr-restore.sh b/deploy/apecloud-mysql/dataprotection/pitr-restore.sh
deleted file mode 100644
index b5e88437299..00000000000
--- a/deploy/apecloud-mysql/dataprotection/pitr-restore.sh
+++ /dev/null
@@ -1,50 +0,0 @@
-#!/bin/bash
-
-BASE_BACKUP_TIME=${BASE_BACKUP_START_TIME}
-if [ -f $DATA_DIR/xtrabackup_info ]; then
- BASE_BACKUP_TIME=$(cat $DATA_DIR/xtrabackup_info | grep start_time | awk -F ' = ' '{print $2}');
- BASE_BACKUP_TIME=$(date -d"${BASE_BACKUP_TIME}" -u '+%Y-%m-%dT%H:%M:%SZ')
-fi
-log_index_name="archive_log.index"
-
-function fetch_pitr_binlogs() {
- cd ${BACKUP_DIR}
- echo "INFO: fetch binlogs from ${BASE_BACKUP_TIME}"
- kb_recovery_timestamp=$(date -d "${KB_RECOVERY_TIME}" +%s)
- for file in $(find . -newermt "${BASE_BACKUP_TIME}" -type f -exec ls -tr {} + | grep .zst );do
- file_path=${file#./}
- file_without_zst=${file_path%.*}
- dir_path=`dirname ${file_path}`
- # mkdir the log directory
- mkdir -p ${PITR_DIR}/${dir_path}
- zstd -d ${file} -o ${PITR_DIR}/${file_without_zst}
- echo "${PITR_RELATIVE_PATH}/${file_without_zst}" >> ${PITR_DIR}/${log_index_name}
- # check if the binlog file contains the data for recovery time
- log_start_time=$(mysqlbinlog ${PITR_DIR}/${file_without_zst} | grep -m 1 "end_log_pos" | awk '{print $1, $2}'|tr -d '#')
- log_start_timestamp=$(date -d "${log_start_time}" +%s)
- if [[ ${log_start_timestamp} -gt ${kb_recovery_timestamp} ]];then
- break
- fi
- done
-}
-
-function save_to_restore_file() {
- if [ -f ${DATA_DIR}/.xtrabackup_restore_new_cluster ];then
- restore_signal_file=${DATA_DIR}/.xtrabackup_restore_new_cluster
- else
- restore_signal_file=${DATA_DIR}/.restore_new_cluster
- fi
- echo "archive_log_index=${PITR_RELATIVE_PATH}/${log_index_name}" > ${restore_signal_file}
- kb_recover_time=$(date -d "${KB_RECOVERY_TIME}" -u '+%Y-%m-%d %H:%M:%S')
- echo "recovery_target_datetime=${kb_recover_time}" >> ${restore_signal_file}
- sync
-}
-
-fetch_pitr_binlogs
-
-if [ -f ${PITR_DIR}/${log_index_name} ];then
- save_to_restore_file
- echo "INFO: fetch binlog finished."
-else
- echo "INFO: didn't get any binlogs."
-fi
diff --git a/deploy/apecloud-mysql/dataprotection/restore.sh b/deploy/apecloud-mysql/dataprotection/restore.sh
new file mode 100644
index 00000000000..89f6b689d74
--- /dev/null
+++ b/deploy/apecloud-mysql/dataprotection/restore.sh
@@ -0,0 +1,12 @@
+#!/bin/bash
+set -e
+mkdir -p ${DATA_DIR}
+TMP_DIR=${DATA_MOUNT_DIR}/temp
+mkdir -p ${TMP_DIR} && cd ${TMP_DIR}
+xbstream -x <${DP_BACKUP_DIR}/${DP_BACKUP_NAME}.xbstream
+xtrabackup --decompress --remove-original --target-dir=${TMP_DIR}
+xtrabackup --prepare --target-dir=${TMP_DIR}
+xtrabackup --move-back --target-dir=${TMP_DIR} --datadir=${DATA_DIR}/ --log-bin=${LOG_BIN}
+touch ${DATA_DIR}/${SIGNAL_FILE}
+rm -rf ${TMP_DIR}
+chmod -R 0777 ${DATA_DIR}
diff --git a/deploy/apecloud-mysql/templates/actionset.yaml b/deploy/apecloud-mysql/templates/actionset.yaml
new file mode 100644
index 00000000000..5d728080701
--- /dev/null
+++ b/deploy/apecloud-mysql/templates/actionset.yaml
@@ -0,0 +1,65 @@
+apiVersion: dataprotection.kubeblocks.io/v1alpha1
+kind: ActionSet
+metadata:
+ name: xtrabackup-for-apecloud-mysql
+ labels:
+ clusterdefinition.kubeblocks.io/name: apecloud-mysql
+spec:
+ backupType: Full
+ env:
+ - name: DATA_DIR
+ value: {{ .Values.mysqlConfigs.dataDir }}
+ - name: LOG_BIN
+ value: {{ .Values.mysqlConfigs.logBin }}
+ - name: DP_DB_PORT
+ value: "3306"
+ - name: DATA_MOUNT_DIR
+ value: {{ .Values.mysqlConfigs.dataMountPath }}
+ - name: SIGNAL_FILE
+ value: .xtrabackup_restore_new_cluster
+ backup:
+ preBackup: []
+ postBackup: []
+ backupData:
+ image: registry.cn-hangzhou.aliyuncs.com/apecloud/apecloud-xtrabackup:latest
+ runOnTargetPodNode: true
+ command:
+ - sh
+ - -c
+ - |
+ {{- .Files.Get "dataprotection/backup.sh" | nindent 8 }}
+ syncProgress:
+ enabled: true
+ intervalSeconds: 5
+ restore:
+ prepareData:
+ image: registry.cn-hangzhou.aliyuncs.com/apecloud/apecloud-xtrabackup:latest
+ command:
+ - sh
+ - -c
+ - |
+ {{- .Files.Get "dataprotection/restore.sh" | nindent 8 }}
+ postReady: []
+---
+apiVersion: dataprotection.kubeblocks.io/v1alpha1
+kind: ActionSet
+metadata:
+ name: volumesnapshot-for-apecloud-mysql
+ labels:
+ clusterdefinition.kubeblocks.io/name: apecloud-mysql
+spec:
+ backupType: Full
+ env:
+ - name: DATA_DIR
+ value: /data/mysql/data
+ - name: SIGNAL_FILE
+ value: .restore_new_cluster
+ backup: {}
+ restore:
+ prepareData:
+ image: registry.cn-hangzhou.aliyuncs.com/apecloud/apecloud-xtrabackup:latest
+ command:
+ - sh
+ - -c
+ - touch {{ .Values.mysqlConfigs.dataDir }}/${SIGNAL_FILE}; sync
+ postReady: []
\ No newline at end of file
diff --git a/deploy/apecloud-mysql/templates/backuppolicytemplate.yaml b/deploy/apecloud-mysql/templates/backuppolicytemplate.yaml
index 06f3d467ac9..462d9b11065 100644
--- a/deploy/apecloud-mysql/templates/backuppolicytemplate.yaml
+++ b/deploy/apecloud-mysql/templates/backuppolicytemplate.yaml
@@ -11,39 +11,30 @@ spec:
clusterDefinitionRef: apecloud-mysql
backupPolicies:
- componentDefRef: mysql
- retention:
- ttl: 7d
- schedule:
- startingDeadlineMinutes: 120
- snapshot:
- enable: false
- cronExpression: "0 18 * * *"
- datafile:
- enable: false
- cronExpression: "0 18 * * *"
- logfile:
- enable: false
- cronExpression: "*/5 * * * *"
- snapshot:
- hooks:
- containerName: mysql
- preCommands:
- - touch {{ .Values.mysqlConfigs.dataDir }}/.restore_new_cluster; sync
- postCommands:
- - "rm -f {{ .Values.mysqlConfigs.dataDir }}/.restore_new_cluster; sync"
- target:
- role: leader
- datafile:
- backupToolName: xtrabackup-for-apecloud-mysql
- backupStatusUpdates:
- - updateStage: post
- useTargetPodServiceAccount: true
- target:
- role: follower
- logfile:
- backupToolName: apecloud-mysql-pitr-tool
- backupStatusUpdates:
- - updateStage: post
- useTargetPodServiceAccount: true
- target:
- role: follower
+ retentionPeriod: 7d
+ target:
+ role: follower
+ backupMethods:
+ - name: xtrabackup
+ snapshotVolumes: false
+ actionSetName: xtrabackup-for-apecloud-mysql
+ targetVolumes:
+ volumeMounts:
+ - name: data
+ mountPath: {{ .Values.mysqlConfigs.dataMountPath }}
+ - name: volume-snapshot
+ snapshotVolumes: true
+ actionSetName: volumesnapshot-for-apecloud-mysql
+ targetVolumes:
+ volumes:
+ - data
+ volumeMounts:
+ - name: data
+ mountPath: {{ .Values.mysqlConfigs.dataMountPath }}
+ schedules:
+ - backupMethod: xtrabackup
+ enabled: false
+ cronExpression: "0 18 * * *"
+ - backupMethod: volume-snapshot
+ enabled: false
+ cronExpression: "0 18 * * *"
\ No newline at end of file
diff --git a/deploy/apecloud-mysql/templates/backuppolicytemplateforhscale.yaml b/deploy/apecloud-mysql/templates/backuppolicytemplateforhscale.yaml
index e8c96578d64..cfb63b15e60 100644
--- a/deploy/apecloud-mysql/templates/backuppolicytemplateforhscale.yaml
+++ b/deploy/apecloud-mysql/templates/backuppolicytemplateforhscale.yaml
@@ -10,14 +10,28 @@ spec:
identifier: hscale
backupPolicies:
- componentDefRef: mysql
- snapshot:
- hooks:
- containerName: mysql
- preCommands:
- - "touch {{ .Values.mysqlConfigs.dataDir }}/.restore; sync"
- postCommands:
- - "rm -f {{ .Values.mysqlConfigs.dataDir }}/.restore; sync"
- target:
- role: leader
- datafile:
- backupToolName: xtrabackup-for-apecloud-mysql-for-hscale
+ target:
+ role: follower
+ backupMethods:
+ - name: volume-snapshot
+ snapshotVolumes: true
+ actionSetName: volumesnapshot-for-apecloud-mysql
+ targetVolumes:
+ volumes:
+ - data
+ volumeMounts:
+ - name: data
+ mountPath: {{ .Values.mysqlConfigs.dataMountPath }}
+ env:
+ - name: SIGNAL_FILE
+ value: .restore
+ - name: xtrabackup
+ snapshotVolumes: false
+ actionSetName: xtrabackup-for-apecloud-mysql
+ targetVolumes:
+ volumeMounts:
+ - name: data
+ mountPath: {{ .Values.mysqlConfigs.dataMountPath }}
+ env:
+ - name: SIGNAL_FILE
+ value: .xtrabackup_restore
\ No newline at end of file
diff --git a/deploy/apecloud-mysql/templates/backuptool-pitr.yaml b/deploy/apecloud-mysql/templates/backuptool-pitr.yaml
deleted file mode 100644
index e50fc0d5112..00000000000
--- a/deploy/apecloud-mysql/templates/backuptool-pitr.yaml
+++ /dev/null
@@ -1,63 +0,0 @@
-apiVersion: dataprotection.kubeblocks.io/v1alpha1
-kind: BackupTool
-metadata:
- labels:
- clusterdefinition.kubeblocks.io/name: apecloud-mysql
- kubeblocks.io/backup-tool-type: pitr
- {{- include "apecloud-mysql.labels" . | nindent 4 }}
- name: apecloud-mysql-pitr-tool
-spec:
- deployKind: statefulSet
- env:
- - name: VOLUME_DATA_DIR
- value: {{ .Values.mysqlConfigs.dataMountPath }}
- - name: DATA_DIR
- value: {{ .Values.mysqlConfigs.dataDir }}
- - name: PITR_RELATIVE_PATH
- value: pitr-logs
- - name: PITR_DIR
- value: "$(DATA_DIR)/$(PITR_RELATIVE_PATH)"
- - name: CONF_DIR
- value: "$(VOLUME_DATA_DIR)/conf"
- - name: TIME_FORMAT
- value: 2006-01-02T15:04:05Z
- - name: DP_TARGET_POD_ROLE
- # TODO input by backup policy
- value: follower
- - name: DP_DB_PORT
- value: "3306"
- - name: DP_INTERVAL_SECONDS
- value: "10"
- - name: FLUSH_BINLOG_INTERVAL_SECONDS
- value: "3600"
- image: apecloud/wal-g:mysql-latest
- logical:
- restoreCommands:
- - bash
- - -c
- - |
- #!/bin/bash
- set -e;
- echo "INFO: waiting for analysis of archive logs to complete."
- while true; do
- if [ ! -f ${DATA_DIR}/.xtrabackup_restore_new_cluster ] && [ ! -f ${DATA_DIR}/.restore_new_cluster ];then
- break
- fi
- sleep 1
- done
- rm -rf ${DATA_DIR}/${PITR_RELATIVE_PATH};
- echo "INFO: remove ${DATA_DIR}/${PITR_RELATIVE_PATH}."
- physical:
- restoreCommands:
- - bash
- - -c
- - |
- set -e;
- {{- .Files.Get "dataprotection/pitr-restore.sh" | nindent 8 }}
- backupCommands:
- - bash
- - -c
- - |
- set -e;
- {{- .Files.Get "dataprotection/pitr-backup.sh" | nindent 6 }}
- type: pitr
diff --git a/deploy/apecloud-mysql/templates/backuptool.yaml b/deploy/apecloud-mysql/templates/backuptool.yaml
deleted file mode 100644
index 647bafb3207..00000000000
--- a/deploy/apecloud-mysql/templates/backuptool.yaml
+++ /dev/null
@@ -1,55 +0,0 @@
-apiVersion: dataprotection.kubeblocks.io/v1alpha1
-kind: BackupTool
-metadata:
- name: xtrabackup-for-apecloud-mysql
- labels:
- clusterdefinition.kubeblocks.io/name: apecloud-mysql
- {{- include "apecloud-mysql.labels" . | nindent 4 }}
-spec:
- image: {{ .Values.image.registry | default "docker.io" }}/apecloud/apecloud-xtrabackup:latest
- deployKind: job
- env:
- - name: DATA_DIR
- value: {{ .Values.mysqlConfigs.dataDir }}
- - name: LOG_BIN
- value: {{ .Values.mysqlConfigs.logBin }}
- - name: DP_DB_PORT
- value: "3306"
- - name: DATA_MOUNT_DIR
- value: {{ .Values.mysqlConfigs.dataMountPath }}
- physical:
- restoreCommands:
- - sh
- - -c
- - |
- set -e;
- mkdir -p ${DATA_DIR}
- TMP_DIR=${DATA_MOUNT_DIR}/temp
- mkdir -p ${TMP_DIR} && cd ${TMP_DIR}
- xbstream -x < ${BACKUP_DIR}/${BACKUP_NAME}.xbstream
- xtrabackup --decompress --remove-original --target-dir=${TMP_DIR}
- xtrabackup --prepare --target-dir=${TMP_DIR}
- xtrabackup --move-back --target-dir=${TMP_DIR} --datadir=${DATA_DIR}/ --log-bin=${LOG_BIN}
- touch ${DATA_DIR}/.xtrabackup_restore_new_cluster
- rm -rf ${TMP_DIR}
- chmod -R 0777 ${DATA_DIR}
- incrementalRestoreCommands: []
- logical:
- restoreCommands: []
- incrementalRestoreCommands: []
- backupCommands:
- - sh
- - -c
- - |
- set -e;
- if [ -d ${BACKUP_DIR} ]; then
- rm -rf ${BACKUP_DIR}
- fi
- mkdir -p ${BACKUP_DIR};
- START_TIME=$(date -u '+%Y-%m-%dT%H:%M:%SZ')
- xtrabackup --compress=zstd --backup --safe-slave-backup --slave-info --stream=xbstream \
- --host=${DB_HOST} --user=${DB_USER} --port=${DP_DB_PORT} --password=${DB_PASSWORD} --datadir=${DATA_DIR} > ${BACKUP_DIR}/${BACKUP_NAME}.xbstream
- STOP_TIME=$(date -u '+%Y-%m-%dT%H:%M:%SZ')
- TOTAL_SIZE=$(du -shx ${BACKUP_DIR}|awk '{print $1}')
- echo "{\"totalSize\":\"$TOTAL_SIZE\",\"manifests\":{\"backupLog\":{\"startTime\":\"${START_TIME}\",\"stopTime\":\"${STOP_TIME}\"},\"backupTool\":{\"uploadTotalSize\":\"${TOTAL_SIZE}\"}}}" > ${BACKUP_DIR}/backup.info
- incrementalBackupCommands: []
diff --git a/deploy/apecloud-mysql/templates/backuptoolforhscale.yaml b/deploy/apecloud-mysql/templates/backuptoolforhscale.yaml
deleted file mode 100644
index 120cd7a7743..00000000000
--- a/deploy/apecloud-mysql/templates/backuptoolforhscale.yaml
+++ /dev/null
@@ -1,46 +0,0 @@
-apiVersion: dataprotection.kubeblocks.io/v1alpha1
-kind: BackupTool
-metadata:
- name: xtrabackup-for-apecloud-mysql-for-hscale
- labels:
- clusterdefinition.kubeblocks.io/name: apecloud-mysql
- {{- include "apecloud-mysql.labels" . | nindent 4 }}
-spec:
- image: registry.cn-hangzhou.aliyuncs.com/apecloud/apecloud-xtrabackup:latest
- deployKind: job
- env:
- - name: DATA_DIR
- value: {{ .Values.mysqlConfigs.dataDir }}
- - name: LOG_BIN
- value: {{ .Values.mysqlConfigs.logBin }}
- - name: DATA_MOUNT_DIR
- value: {{ .Values.mysqlConfigs.dataMountPath }}
- physical:
- restoreCommands:
- - sh
- - -c
- - |
- set -e;
- mkdir -p ${DATA_DIR}
- TMP_DIR=${DATA_MOUNT_DIR}/temp
- mkdir -p ${TMP_DIR} && cd ${TMP_DIR}
- xbstream -x < ${BACKUP_DIR}/${BACKUP_NAME}.xbstream
- xtrabackup --decompress --target-dir=${TMP_DIR}
- xtrabackup --prepare --target-dir=${TMP_DIR}
- find . -name "*.qp"|xargs rm -f
- xtrabackup --move-back --target-dir=${TMP_DIR} --datadir=${DATA_DIR}/ --log-bin=${LOG_BIN}
- touch ${DATA_DIR}/.xtrabackup_restore
- rm -rf ${TMP_DIR}
- chmod -R 0777 ${DATA_DIR}
- incrementalRestoreCommands: []
- logical:
- restoreCommands: []
- incrementalRestoreCommands: []
- backupCommands:
- - sh
- - -c
- - |
- set -e
- mkdir -p ${BACKUP_DIR}
- xtrabackup --compress --backup --safe-slave-backup --slave-info --stream=xbstream --host=${DB_HOST} --user=${DB_USER} --password=${DB_PASSWORD} --datadir=${DATA_DIR} > ${BACKUP_DIR}/${BACKUP_NAME}.xbstream
- incrementalBackupCommands: []
diff --git a/deploy/helm/config/rbac/role.yaml b/deploy/helm/config/rbac/role.yaml
index e43058fbb5a..57454b4073c 100644
--- a/deploy/helm/config/rbac/role.yaml
+++ b/deploy/helm/config/rbac/role.yaml
@@ -2,7 +2,6 @@
apiVersion: rbac.authorization.k8s.io/v1
kind: ClusterRole
metadata:
- creationTimestamp: null
name: manager-role
rules:
- apiGroups:
@@ -567,6 +566,32 @@ rules:
- services/status
verbs:
- get
+- apiGroups:
+ - dataprotection.kubeblocks.io
+ resources:
+ - actionsets
+ verbs:
+ - create
+ - delete
+ - get
+ - list
+ - patch
+ - update
+ - watch
+- apiGroups:
+ - dataprotection.kubeblocks.io
+ resources:
+ - actionsets/finalizers
+ verbs:
+ - update
+- apiGroups:
+ - dataprotection.kubeblocks.io
+ resources:
+ - actionsets/status
+ verbs:
+ - get
+ - patch
+ - update
- apiGroups:
- dataprotection.kubeblocks.io
resources:
@@ -650,7 +675,7 @@ rules:
- apiGroups:
- dataprotection.kubeblocks.io
resources:
- - backuptools
+ - backupschedules
verbs:
- create
- delete
@@ -662,13 +687,13 @@ rules:
- apiGroups:
- dataprotection.kubeblocks.io
resources:
- - backuptools/finalizers
+ - backupschedules/finalizers
verbs:
- update
- apiGroups:
- dataprotection.kubeblocks.io
resources:
- - backuptools/status
+ - backupschedules/status
verbs:
- get
- patch
@@ -676,7 +701,7 @@ rules:
- apiGroups:
- dataprotection.kubeblocks.io
resources:
- - restorejobs
+ - restores
verbs:
- create
- delete
@@ -688,13 +713,13 @@ rules:
- apiGroups:
- dataprotection.kubeblocks.io
resources:
- - restorejobs/finalizers
+ - restores/finalizers
verbs:
- update
- apiGroups:
- dataprotection.kubeblocks.io
resources:
- - restorejobs/status
+ - restores/status
verbs:
- get
- patch
diff --git a/deploy/helm/crds/apps.kubeblocks.io_backuppolicytemplates.yaml b/deploy/helm/crds/apps.kubeblocks.io_backuppolicytemplates.yaml
index ba3e79f8e16..2073e6f7e8e 100644
--- a/deploy/helm/crds/apps.kubeblocks.io_backuppolicytemplates.yaml
+++ b/deploy/helm/crds/apps.kubeblocks.io_backuppolicytemplates.yaml
@@ -2,8 +2,7 @@ apiVersion: apiextensions.k8s.io/v1
kind: CustomResourceDefinition
metadata:
annotations:
- controller-gen.kubebuilder.io/version: v0.9.0
- creationTimestamp: null
+ controller-gen.kubebuilder.io/version: v0.12.1
labels:
app.kubernetes.io/name: kubeblocks
name: backuppolicytemplates.apps.kubeblocks.io
@@ -54,6 +53,268 @@ spec:
the specified componentDefinition.
items:
properties:
+ backupMethods:
+ description: backupMethods defines the backup methods.
+ items:
+ description: BackupMethod defines the backup method.
+ properties:
+ actionSetName:
+ description: actionSetName refers to the ActionSet object
+ that defines the backup actions. For volume snapshot
+ backup, the actionSet is not required, the controller
+ will use the CSI volume snapshotter to create the snapshot.
+ type: string
+ env:
+ description: env specifies the environment variables for
+ the backup workload.
+ items:
+ description: EnvVar represents an environment variable
+ present in a Container.
+ properties:
+ name:
+ description: Name of the environment variable. Must
+ be a C_IDENTIFIER.
+ type: string
+ value:
+ description: 'Variable references $(VAR_NAME) are
+ expanded using the previously defined environment
+ variables in the container and any service environment
+ variables. If a variable cannot be resolved, the
+ reference in the input string will be unchanged.
+ Double $$ are reduced to a single $, which allows
+ for escaping the $(VAR_NAME) syntax: i.e. "$$(VAR_NAME)"
+ will produce the string literal "$(VAR_NAME)".
+ Escaped references will never be expanded, regardless
+ of whether the variable exists or not. Defaults
+ to "".'
+ type: string
+ valueFrom:
+ description: Source for the environment variable's
+ value. Cannot be used if value is not empty.
+ properties:
+ configMapKeyRef:
+ description: Selects a key of a ConfigMap.
+ properties:
+ key:
+ description: The key to select.
+ type: string
+ name:
+ description: 'Name of the referent. More
+ info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names
+ TODO: Add other useful fields. apiVersion,
+ kind, uid?'
+ type: string
+ optional:
+ description: Specify whether the ConfigMap
+ or its key must be defined
+ type: boolean
+ required:
+ - key
+ type: object
+ x-kubernetes-map-type: atomic
+ fieldRef:
+ description: 'Selects a field of the pod: supports
+ metadata.name, metadata.namespace, `metadata.labels['''']`,
+ `metadata.annotations['''']`, spec.nodeName,
+ spec.serviceAccountName, status.hostIP, status.podIP,
+ status.podIPs.'
+ properties:
+ apiVersion:
+ description: Version of the schema the FieldPath
+ is written in terms of, defaults to "v1".
+ type: string
+ fieldPath:
+ description: Path of the field to select
+ in the specified API version.
+ type: string
+ required:
+ - fieldPath
+ type: object
+ x-kubernetes-map-type: atomic
+ resourceFieldRef:
+ description: 'Selects a resource of the container:
+ only resources limits and requests (limits.cpu,
+ limits.memory, limits.ephemeral-storage, requests.cpu,
+ requests.memory and requests.ephemeral-storage)
+ are currently supported.'
+ properties:
+ containerName:
+ description: 'Container name: required for
+ volumes, optional for env vars'
+ type: string
+ divisor:
+ anyOf:
+ - type: integer
+ - type: string
+ description: Specifies the output format
+ of the exposed resources, defaults to
+ "1"
+ pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$
+ x-kubernetes-int-or-string: true
+ resource:
+ description: 'Required: resource to select'
+ type: string
+ required:
+ - resource
+ type: object
+ x-kubernetes-map-type: atomic
+ secretKeyRef:
+ description: Selects a key of a secret in the
+ pod's namespace
+ properties:
+ key:
+ description: The key of the secret to select
+ from. Must be a valid secret key.
+ type: string
+ name:
+ description: 'Name of the referent. More
+ info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names
+ TODO: Add other useful fields. apiVersion,
+ kind, uid?'
+ type: string
+ optional:
+ description: Specify whether the Secret
+ or its key must be defined
+ type: boolean
+ required:
+ - key
+ type: object
+ x-kubernetes-map-type: atomic
+ type: object
+ required:
+ - name
+ type: object
+ type: array
+ name:
+ description: the name of backup method.
+ pattern: ^[a-z0-9]([a-z0-9\.\-]*[a-z0-9])?$
+ type: string
+ runtimeSettings:
+ description: runtimeSettings specifies runtime settings
+ for the backup workload container.
+ properties:
+ resources:
+ description: 'resources specifies the resource required
+ by container. More info: https://kubernetes.io/docs/concepts/configuration/manage-compute-resources-container/'
+ properties:
+ claims:
+ description: "Claims lists the names of resources,
+ defined in spec.resourceClaims, that are used
+ by this container. \n This is an alpha field
+ and requires enabling the DynamicResourceAllocation
+ feature gate. \n This field is immutable. It
+ can only be set for containers."
+ items:
+ description: ResourceClaim references one entry
+ in PodSpec.ResourceClaims.
+ properties:
+ name:
+ description: Name must match the name of
+ one entry in pod.spec.resourceClaims of
+ the Pod where this field is used. It makes
+ that resource available inside a container.
+ type: string
+ required:
+ - name
+ type: object
+ type: array
+ x-kubernetes-list-map-keys:
+ - name
+ x-kubernetes-list-type: map
+ limits:
+ additionalProperties:
+ anyOf:
+ - type: integer
+ - type: string
+ pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$
+ x-kubernetes-int-or-string: true
+ description: 'Limits describes the maximum amount
+ of compute resources allowed. More info: https://kubernetes.io/docs/concepts/configuration/manage-resources-containers/'
+ type: object
+ requests:
+ additionalProperties:
+ anyOf:
+ - type: integer
+ - type: string
+ pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$
+ x-kubernetes-int-or-string: true
+ description: 'Requests describes the minimum amount
+ of compute resources required. If Requests is
+ omitted for a container, it defaults to Limits
+ if that is explicitly specified, otherwise to
+ an implementation-defined value. Requests cannot
+ exceed Limits. More info: https://kubernetes.io/docs/concepts/configuration/manage-resources-containers/'
+ type: object
+ type: object
+ type: object
+ snapshotVolumes:
+ default: false
+ description: snapshotVolumes specifies whether to take
+ snapshots of persistent volumes. if true, the BackupScript
+ is not required, the controller will use the CSI volume
+ snapshotter to create the snapshot.
+ type: boolean
+ targetVolumes:
+ description: targetVolumes specifies which volumes from
+ the target should be mounted in the backup workload.
+ properties:
+ volumeMounts:
+ description: volumeMounts specifies the mount for
+ the volumes specified in `Volumes` section.
+ items:
+ description: VolumeMount describes a mounting of
+ a Volume within a container.
+ properties:
+ mountPath:
+ description: Path within the container at which
+ the volume should be mounted. Must not contain
+ ':'.
+ type: string
+ mountPropagation:
+ description: mountPropagation determines how
+ mounts are propagated from the host to container
+ and the other way around. When not set, MountPropagationNone
+ is used. This field is beta in 1.10.
+ type: string
+ name:
+ description: This must match the Name of a Volume.
+ type: string
+ readOnly:
+ description: Mounted read-only if true, read-write
+ otherwise (false or unspecified). Defaults
+ to false.
+ type: boolean
+ subPath:
+ description: Path within the volume from which
+ the container's volume should be mounted.
+ Defaults to "" (volume's root).
+ type: string
+ subPathExpr:
+ description: Expanded path within the volume
+ from which the container's volume should be
+ mounted. Behaves similarly to SubPath but
+ environment variable references $(VAR_NAME)
+ are expanded using the container's environment.
+ Defaults to "" (volume's root). SubPathExpr
+ and SubPath are mutually exclusive.
+ type: string
+ required:
+ - mountPath
+ - name
+ type: object
+ type: array
+ volumes:
+ description: Volumes indicates the list of volumes
+ of targeted application that should be mounted on
+ the backup job.
+ items:
+ type: string
+ type: array
+ type: object
+ required:
+ - name
+ type: object
+ type: array
componentDefRef:
description: componentDefRef references componentDef defined
in ClusterDefinition spec. Need to comply with IANA Service
@@ -61,378 +322,86 @@ spec:
maxLength: 22
pattern: ^[a-z]([a-z0-9\-]*[a-z0-9])?$
type: string
- datafile:
- description: the policy for datafile backup.
+ retentionPeriod:
+ default: 7d
+ description: "retentionPeriod determines a duration up to which
+ the backup should be kept. controller will remove all backups
+ that are older than the RetentionPeriod. For example, RetentionPeriod
+ of `30d` will keep only the backups of last 30 days. Sample
+ duration format: - years: \t2y - months: \t6mo - days: \t\t30d
+ - hours: \t12h - minutes: \t30m You can also combine the above
+ durations. For example: 30d12h30m"
+ type: string
+ schedules:
+ description: schedule policy for backup.
+ items:
+ properties:
+ backupMethod:
+ description: backupMethod specifies the backup method
+ name that is defined in backupPolicy.
+ type: string
+ cronExpression:
+ description: the cron expression for schedule, the timezone
+ is in UTC. see https://en.wikipedia.org/wiki/Cron.
+ type: string
+ enabled:
+ description: enabled specifies whether the backup schedule
+ is enabled or not.
+ type: boolean
+ required:
+ - backupMethod
+ - cronExpression
+ type: object
+ type: array
+ target:
+ description: target instance for backup.
properties:
- backupStatusUpdates:
- description: define how to update metadata for backup status.
- items:
- properties:
- containerName:
- description: which container name that kubectl can
- execute.
- type: string
- path:
- description: 'specify the json path of backup object
- for patch. example: manifests.backupLog -- means
- patch the backup json path of status.manifests.backupLog.'
- type: string
- script:
- description: the shell Script commands to collect
- backup status metadata. The script must exist in
- the container of ContainerName and the output format
- must be set to JSON. Note that outputting to stderr
- may cause the result format to not be in JSON.
- type: string
- updateStage:
- description: 'when to update the backup status, pre:
- before backup, post: after backup'
- enum:
- - pre
- - post
- type: string
- useTargetPodServiceAccount:
- description: useTargetPodServiceAccount defines whether
- this job requires the service account of the backup
- target pod. if true, will use the service account
- of the backup target pod. otherwise, will use the
- system service account.
- type: boolean
- required:
- - updateStage
- type: object
- type: array
- backupToolName:
- description: which backup tool to perform database backup,
- only support one tool.
- pattern: ^[a-z0-9]([a-z0-9\.\-]*[a-z0-9])?$
+ account:
+ description: refer to spec.componentDef.systemAccounts.accounts[*].name
+ in ClusterDefinition. the secret created by this account
+ will be used to connect the database. if not set, the
+ secret created by spec.ConnectionCredential of the ClusterDefinition
+ will be used. it will be transformed to a secret for BackupPolicy's
+ target secret.
type: string
- backupsHistoryLimit:
- default: 7
- description: the number of automatic backups to retain.
- Value must be non-negative integer. 0 means NO limit on
- the number of backups.
- format: int32
- type: integer
- onFailAttempted:
- description: count of backup stop retries on fail.
- format: int32
- type: integer
- target:
- description: target instance for backup.
+ connectionCredentialKey:
+ description: connectionCredentialKey defines connection
+ credential key in secret which created by spec.ConnectionCredential
+ of the ClusterDefinition. it will be ignored when "account"
+ is set.
properties:
- account:
- description: refer to spec.componentDef.systemAccounts.accounts[*].name
- in ClusterDefinition. the secret created by this account
- will be used to connect the database. if not set,
- the secret created by spec.ConnectionCredential of
- the ClusterDefinition will be used. it will be transformed
- to a secret for BackupPolicy's target secret.
+ hostKey:
+ description: hostKey specifies the map key of the host
+ in the connection credential secret.
type: string
- connectionCredentialKey:
- description: connectionCredentialKey defines connection
- credential key in secret which created by spec.ConnectionCredential
- of the ClusterDefinition. it will be ignored when
- "account" is set.
- properties:
- passwordKey:
- description: the key of password in the ConnectionCredential
- secret. if not set, the default key is "password".
- type: string
- usernameKey:
- description: the key of username in the ConnectionCredential
- secret. if not set, the default key is "username".
- type: string
- type: object
- role:
- description: 'select instance of corresponding role
- for backup, role are: - the name of Leader/Follower/Leaner
- for Consensus component. - primary or secondary for
- Replication component. finally, invalid role of the
- component will be ignored. such as if workload type
- is Replication and component''s replicas is 1, the
- secondary role is invalid. and it also will be ignored
- when component is Stateful/Stateless. the role will
- be transformed to a role LabelSelector for BackupPolicy''s
- target attribute.'
+ passwordKey:
+ description: the key of password in the ConnectionCredential
+ secret. if not set, the default key is "password".
type: string
- type: object
- type: object
- logfile:
- description: the policy for logfile backup.
- properties:
- backupStatusUpdates:
- description: define how to update metadata for backup status.
- items:
- properties:
- containerName:
- description: which container name that kubectl can
- execute.
- type: string
- path:
- description: 'specify the json path of backup object
- for patch. example: manifests.backupLog -- means
- patch the backup json path of status.manifests.backupLog.'
- type: string
- script:
- description: the shell Script commands to collect
- backup status metadata. The script must exist in
- the container of ContainerName and the output format
- must be set to JSON. Note that outputting to stderr
- may cause the result format to not be in JSON.
- type: string
- updateStage:
- description: 'when to update the backup status, pre:
- before backup, post: after backup'
- enum:
- - pre
- - post
- type: string
- useTargetPodServiceAccount:
- description: useTargetPodServiceAccount defines whether
- this job requires the service account of the backup
- target pod. if true, will use the service account
- of the backup target pod. otherwise, will use the
- system service account.
- type: boolean
- required:
- - updateStage
- type: object
- type: array
- backupToolName:
- description: which backup tool to perform database backup,
- only support one tool.
- pattern: ^[a-z0-9]([a-z0-9\.\-]*[a-z0-9])?$
- type: string
- backupsHistoryLimit:
- default: 7
- description: the number of automatic backups to retain.
- Value must be non-negative integer. 0 means NO limit on
- the number of backups.
- format: int32
- type: integer
- onFailAttempted:
- description: count of backup stop retries on fail.
- format: int32
- type: integer
- target:
- description: target instance for backup.
- properties:
- account:
- description: refer to spec.componentDef.systemAccounts.accounts[*].name
- in ClusterDefinition. the secret created by this account
- will be used to connect the database. if not set,
- the secret created by spec.ConnectionCredential of
- the ClusterDefinition will be used. it will be transformed
- to a secret for BackupPolicy's target secret.
+ portKey:
+ default: port
+ description: portKey specifies the map key of the port
+ in the connection credential secret.
type: string
- connectionCredentialKey:
- description: connectionCredentialKey defines connection
- credential key in secret which created by spec.ConnectionCredential
- of the ClusterDefinition. it will be ignored when
- "account" is set.
- properties:
- passwordKey:
- description: the key of password in the ConnectionCredential
- secret. if not set, the default key is "password".
- type: string
- usernameKey:
- description: the key of username in the ConnectionCredential
- secret. if not set, the default key is "username".
- type: string
- type: object
- role:
- description: 'select instance of corresponding role
- for backup, role are: - the name of Leader/Follower/Leaner
- for Consensus component. - primary or secondary for
- Replication component. finally, invalid role of the
- component will be ignored. such as if workload type
- is Replication and component''s replicas is 1, the
- secondary role is invalid. and it also will be ignored
- when component is Stateful/Stateless. the role will
- be transformed to a role LabelSelector for BackupPolicy''s
- target attribute.'
+ usernameKey:
+ description: the key of username in the ConnectionCredential
+ secret. if not set, the default key is "username".
type: string
type: object
- type: object
- retention:
- description: retention describe how long the Backup should be
- retained. if not set, will be retained forever.
- properties:
- ttl:
- description: ttl is a time string ending with the 'd'|'D'|'h'|'H'
- character to describe how long the Backup should be retained.
- if not set, will be retained forever.
- pattern: ^\d+[d|D|h|H]$
+ role:
+ description: 'select instance of corresponding role for
+ backup, role are: - the name of Leader/Follower/Leaner
+ for Consensus component. - primary or secondary for Replication
+ component. finally, invalid role of the component will
+ be ignored. such as if workload type is Replication and
+ component''s replicas is 1, the secondary role is invalid.
+ and it also will be ignored when component is Stateful/Stateless.
+ the role will be transformed to a role LabelSelector for
+ BackupPolicy''s target attribute.'
type: string
type: object
- schedule:
- description: schedule policy for backup.
- properties:
- datafile:
- description: schedule policy for datafile backup.
- properties:
- cronExpression:
- description: the cron expression for schedule, the timezone
- is in UTC. see https://en.wikipedia.org/wiki/Cron.
- type: string
- enable:
- description: enable or disable the schedule.
- type: boolean
- required:
- - cronExpression
- - enable
- type: object
- logfile:
- description: schedule policy for logfile backup.
- properties:
- cronExpression:
- description: the cron expression for schedule, the timezone
- is in UTC. see https://en.wikipedia.org/wiki/Cron.
- type: string
- enable:
- description: enable or disable the schedule.
- type: boolean
- required:
- - cronExpression
- - enable
- type: object
- snapshot:
- description: schedule policy for snapshot backup.
- properties:
- cronExpression:
- description: the cron expression for schedule, the timezone
- is in UTC. see https://en.wikipedia.org/wiki/Cron.
- type: string
- enable:
- description: enable or disable the schedule.
- type: boolean
- required:
- - cronExpression
- - enable
- type: object
- startingDeadlineMinutes:
- description: startingDeadlineMinutes defines the deadline
- in minutes for starting the backup job if it misses scheduled
- time for any reason.
- format: int64
- maximum: 1440
- minimum: 0
- type: integer
- type: object
- snapshot:
- description: the policy for snapshot backup.
- properties:
- backupStatusUpdates:
- description: define how to update metadata for backup status.
- items:
- properties:
- containerName:
- description: which container name that kubectl can
- execute.
- type: string
- path:
- description: 'specify the json path of backup object
- for patch. example: manifests.backupLog -- means
- patch the backup json path of status.manifests.backupLog.'
- type: string
- script:
- description: the shell Script commands to collect
- backup status metadata. The script must exist in
- the container of ContainerName and the output format
- must be set to JSON. Note that outputting to stderr
- may cause the result format to not be in JSON.
- type: string
- updateStage:
- description: 'when to update the backup status, pre:
- before backup, post: after backup'
- enum:
- - pre
- - post
- type: string
- useTargetPodServiceAccount:
- description: useTargetPodServiceAccount defines whether
- this job requires the service account of the backup
- target pod. if true, will use the service account
- of the backup target pod. otherwise, will use the
- system service account.
- type: boolean
- required:
- - updateStage
- type: object
- type: array
- backupsHistoryLimit:
- default: 7
- description: the number of automatic backups to retain.
- Value must be non-negative integer. 0 means NO limit on
- the number of backups.
- format: int32
- type: integer
- hooks:
- description: execute hook commands for backup.
- properties:
- containerName:
- description: which container can exec command
- type: string
- image:
- description: exec command with image
- type: string
- postCommands:
- description: post backup to perform commands
- items:
- type: string
- type: array
- preCommands:
- description: pre backup to perform commands
- items:
- type: string
- type: array
- type: object
- onFailAttempted:
- description: count of backup stop retries on fail.
- format: int32
- type: integer
- target:
- description: target instance for backup.
- properties:
- account:
- description: refer to spec.componentDef.systemAccounts.accounts[*].name
- in ClusterDefinition. the secret created by this account
- will be used to connect the database. if not set,
- the secret created by spec.ConnectionCredential of
- the ClusterDefinition will be used. it will be transformed
- to a secret for BackupPolicy's target secret.
- type: string
- connectionCredentialKey:
- description: connectionCredentialKey defines connection
- credential key in secret which created by spec.ConnectionCredential
- of the ClusterDefinition. it will be ignored when
- "account" is set.
- properties:
- passwordKey:
- description: the key of password in the ConnectionCredential
- secret. if not set, the default key is "password".
- type: string
- usernameKey:
- description: the key of username in the ConnectionCredential
- secret. if not set, the default key is "username".
- type: string
- type: object
- role:
- description: 'select instance of corresponding role
- for backup, role are: - the name of Leader/Follower/Leaner
- for Consensus component. - primary or secondary for
- Replication component. finally, invalid role of the
- component will be ignored. such as if workload type
- is Replication and component''s replicas is 1, the
- secondary role is invalid. and it also will be ignored
- when component is Stateful/Stateless. the role will
- be transformed to a role LabelSelector for BackupPolicy''s
- target attribute.'
- type: string
- type: object
- type: object
required:
+ - backupMethods
- componentDefRef
type: object
minItems: 1
diff --git a/deploy/helm/crds/apps.kubeblocks.io_clusterdefinitions.yaml b/deploy/helm/crds/apps.kubeblocks.io_clusterdefinitions.yaml
index a86c3b22a9e..538dc69bc71 100644
--- a/deploy/helm/crds/apps.kubeblocks.io_clusterdefinitions.yaml
+++ b/deploy/helm/crds/apps.kubeblocks.io_clusterdefinitions.yaml
@@ -2,8 +2,7 @@ apiVersion: apiextensions.k8s.io/v1
kind: CustomResourceDefinition
metadata:
annotations:
- controller-gen.kubebuilder.io/version: v0.9.0
- creationTimestamp: null
+ controller-gen.kubebuilder.io/version: v0.12.1
labels:
app.kubernetes.io/name: kubeblocks
name: clusterdefinitions.apps.kubeblocks.io
@@ -700,6 +699,7 @@ spec:
type: object
type: array
type: object
+ x-kubernetes-map-type: atomic
weight:
description: Weight associated with matching
the corresponding nodeSelectorTerm, in the
@@ -810,10 +810,12 @@ spec:
type: object
type: array
type: object
+ x-kubernetes-map-type: atomic
type: array
required:
- nodeSelectorTerms
type: object
+ x-kubernetes-map-type: atomic
type: object
podAffinity:
description: Describes pod affinity scheduling rules
@@ -900,6 +902,7 @@ spec:
ANDed.
type: object
type: object
+ x-kubernetes-map-type: atomic
namespaceSelector:
description: A label query over the set
of namespaces that the term applies
@@ -963,6 +966,7 @@ spec:
ANDed.
type: object
type: object
+ x-kubernetes-map-type: atomic
namespaces:
description: namespaces specifies a static
list of namespace names that the term
@@ -1073,6 +1077,7 @@ spec:
only "value". The requirements are ANDed.
type: object
type: object
+ x-kubernetes-map-type: atomic
namespaceSelector:
description: A label query over the set of
namespaces that the term applies to. The
@@ -1131,6 +1136,7 @@ spec:
only "value". The requirements are ANDed.
type: object
type: object
+ x-kubernetes-map-type: atomic
namespaces:
description: namespaces specifies a static
list of namespace names that the term applies
@@ -1243,6 +1249,7 @@ spec:
ANDed.
type: object
type: object
+ x-kubernetes-map-type: atomic
namespaceSelector:
description: A label query over the set
of namespaces that the term applies
@@ -1306,6 +1313,7 @@ spec:
ANDed.
type: object
type: object
+ x-kubernetes-map-type: atomic
namespaces:
description: namespaces specifies a static
list of namespace names that the term
@@ -1417,6 +1425,7 @@ spec:
only "value". The requirements are ANDed.
type: object
type: object
+ x-kubernetes-map-type: atomic
namespaceSelector:
description: A label query over the set of
namespaces that the term applies to. The
@@ -1475,6 +1484,7 @@ spec:
only "value". The requirements are ANDed.
type: object
type: object
+ x-kubernetes-map-type: atomic
namespaces:
description: namespaces specifies a static
list of namespace names that the term applies
@@ -1594,6 +1604,7 @@ spec:
required:
- key
type: object
+ x-kubernetes-map-type: atomic
fieldRef:
description: 'Selects a field of the pod:
supports metadata.name, metadata.namespace,
@@ -1613,6 +1624,7 @@ spec:
required:
- fieldPath
type: object
+ x-kubernetes-map-type: atomic
resourceFieldRef:
description: 'Selects a resource of the
container: only resources limits and requests
@@ -1640,6 +1652,7 @@ spec:
required:
- resource
type: object
+ x-kubernetes-map-type: atomic
secretKeyRef:
description: Selects a key of a secret in
the pod's namespace
@@ -1662,6 +1675,7 @@ spec:
required:
- key
type: object
+ x-kubernetes-map-type: atomic
type: object
required:
- name
@@ -1694,6 +1708,7 @@ spec:
must be defined
type: boolean
type: object
+ x-kubernetes-map-type: atomic
prefix:
description: An optional identifier to prepend
to each key in the ConfigMap. Must be a C_IDENTIFIER.
@@ -1712,6 +1727,7 @@ spec:
must be defined
type: boolean
type: object
+ x-kubernetes-map-type: atomic
type: object
type: array
image:
@@ -1776,7 +1792,11 @@ spec:
custom header to be used in HTTP probes
properties:
name:
- description: The header field name
+ description: The header field name.
+ This will be canonicalized upon
+ output, so case-variant names
+ will be understood as the same
+ header.
type: string
value:
description: The header field value
@@ -1884,7 +1904,11 @@ spec:
custom header to be used in HTTP probes
properties:
name:
- description: The header field name
+ description: The header field name.
+ This will be canonicalized upon
+ output, so case-variant names
+ will be understood as the same
+ header.
type: string
value:
description: The header field value
@@ -1971,8 +1995,7 @@ spec:
type: integer
grpc:
description: GRPC specifies an action involving
- a GRPC port. This is a beta field and requires
- enabling GRPCContainerProbe feature gate.
+ a GRPC port.
properties:
port:
description: Port number of the gRPC service.
@@ -2006,7 +2029,10 @@ spec:
header to be used in HTTP probes
properties:
name:
- description: The header field name
+ description: The header field name.
+ This will be canonicalized upon output,
+ so case-variant names will be understood
+ as the same header.
type: string
value:
description: The header field value
@@ -2188,8 +2214,7 @@ spec:
type: integer
grpc:
description: GRPC specifies an action involving
- a GRPC port. This is a beta field and requires
- enabling GRPCContainerProbe feature gate.
+ a GRPC port.
properties:
port:
description: Port number of the gRPC service.
@@ -2223,7 +2248,10 @@ spec:
header to be used in HTTP probes
properties:
name:
- description: The header field name
+ description: The header field name.
+ This will be canonicalized upon output,
+ so case-variant names will be understood
+ as the same header.
type: string
value:
description: The header field value
@@ -2318,6 +2346,28 @@ spec:
format: int32
type: integer
type: object
+ resizePolicy:
+ description: Resources resize policy for the container.
+ items:
+ description: ContainerResizePolicy represents resource
+ resize policy for the container.
+ properties:
+ resourceName:
+ description: 'Name of the resource to which
+ this resource resize policy applies. Supported
+ values: cpu, memory.'
+ type: string
+ restartPolicy:
+ description: Restart policy to apply when specified
+ resource is resized. If not specified, it
+ defaults to NotRequired.
+ type: string
+ required:
+ - resourceName
+ - restartPolicy
+ type: object
+ type: array
+ x-kubernetes-list-type: atomic
resources:
description: 'Compute Resources required by this container.
Cannot be updated. More info: https://kubernetes.io/docs/concepts/configuration/manage-resources-containers/'
@@ -2367,10 +2417,33 @@ spec:
of compute resources required. If Requests is
omitted for a container, it defaults to Limits
if that is explicitly specified, otherwise to
- an implementation-defined value. More info:
- https://kubernetes.io/docs/concepts/configuration/manage-resources-containers/'
+ an implementation-defined value. Requests cannot
+ exceed Limits. More info: https://kubernetes.io/docs/concepts/configuration/manage-resources-containers/'
type: object
type: object
+ restartPolicy:
+ description: 'RestartPolicy defines the restart behavior
+ of individual containers in a pod. This field may
+ only be set for init containers, and the only allowed
+ value is "Always". For non-init containers or when
+ this field is not specified, the restart behavior
+ is defined by the Pod''s restart policy and the
+ container type. Setting the RestartPolicy as "Always"
+ for the init container will have the following effect:
+ this init container will be continually restarted
+ on exit until all regular containers have terminated.
+ Once all regular containers have completed, all
+ init containers with restartPolicy "Always" will
+ be shut down. This lifecycle differs from normal
+ init containers and is often referred to as a "sidecar"
+ container. Although this init container still starts
+ in the init container sequence, it does not wait
+ for the container to complete before proceeding
+ to the next init container. Instead, the next init
+ container starts immediately after this init container
+ is started, or after any startupProbe has successfully
+ completed.'
+ type: string
securityContext:
description: 'SecurityContext defines the security
options the container should be run with. If set,
@@ -2506,8 +2579,9 @@ spec:
be used. The profile must be preconfigured
on the node to work. Must be a descending
path, relative to the kubelet's configured
- seccomp profile location. Must only be set
- if type is "Localhost".
+ seccomp profile location. Must be set if
+ type is "Localhost". Must NOT be set for
+ any other type.
type: string
type:
description: "type indicates which kind of
@@ -2544,17 +2618,12 @@ spec:
hostProcess:
description: HostProcess determines if a container
should be run as a 'Host Process' container.
- This field is alpha-level and will only
- be honored by components that enable the
- WindowsHostProcessContainers feature flag.
- Setting this field without the feature flag
- will result in errors when validating the
- Pod. All of a Pod's containers must have
- the same effective HostProcess value (it
- is not allowed to have a mix of HostProcess
- containers and non-HostProcess containers). In
- addition, if HostProcess is true then HostNetwork
- must also be set to true.
+ All of a Pod's containers must have the
+ same effective HostProcess value (it is
+ not allowed to have a mix of HostProcess
+ containers and non-HostProcess containers).
+ In addition, if HostProcess is true then
+ HostNetwork must also be set to true.
type: boolean
runAsUserName:
description: The UserName in Windows to run
@@ -2605,8 +2674,7 @@ spec:
type: integer
grpc:
description: GRPC specifies an action involving
- a GRPC port. This is a beta field and requires
- enabling GRPCContainerProbe feature gate.
+ a GRPC port.
properties:
port:
description: Port number of the gRPC service.
@@ -2640,7 +2708,10 @@ spec:
header to be used in HTTP probes
properties:
name:
- description: The header field name
+ description: The header field name.
+ This will be canonicalized upon output,
+ so case-variant names will be understood
+ as the same header.
type: string
value:
description: The header field value
@@ -3009,6 +3080,7 @@ spec:
required:
- key
type: object
+ x-kubernetes-map-type: atomic
fieldRef:
description: 'Selects a field of the pod:
supports metadata.name, metadata.namespace,
@@ -3028,6 +3100,7 @@ spec:
required:
- fieldPath
type: object
+ x-kubernetes-map-type: atomic
resourceFieldRef:
description: 'Selects a resource of the
container: only resources limits and requests
@@ -3055,6 +3128,7 @@ spec:
required:
- resource
type: object
+ x-kubernetes-map-type: atomic
secretKeyRef:
description: Selects a key of a secret in
the pod's namespace
@@ -3077,6 +3151,7 @@ spec:
required:
- key
type: object
+ x-kubernetes-map-type: atomic
type: object
required:
- name
@@ -3109,6 +3184,7 @@ spec:
must be defined
type: boolean
type: object
+ x-kubernetes-map-type: atomic
prefix:
description: An optional identifier to prepend
to each key in the ConfigMap. Must be a C_IDENTIFIER.
@@ -3127,6 +3203,7 @@ spec:
must be defined
type: boolean
type: object
+ x-kubernetes-map-type: atomic
type: object
type: array
image:
@@ -3187,7 +3264,11 @@ spec:
custom header to be used in HTTP probes
properties:
name:
- description: The header field name
+ description: The header field name.
+ This will be canonicalized upon
+ output, so case-variant names
+ will be understood as the same
+ header.
type: string
value:
description: The header field value
@@ -3295,7 +3376,11 @@ spec:
custom header to be used in HTTP probes
properties:
name:
- description: The header field name
+ description: The header field name.
+ This will be canonicalized upon
+ output, so case-variant names
+ will be understood as the same
+ header.
type: string
value:
description: The header field value
@@ -3381,8 +3466,7 @@ spec:
type: integer
grpc:
description: GRPC specifies an action involving
- a GRPC port. This is a beta field and requires
- enabling GRPCContainerProbe feature gate.
+ a GRPC port.
properties:
port:
description: Port number of the gRPC service.
@@ -3416,7 +3500,10 @@ spec:
header to be used in HTTP probes
properties:
name:
- description: The header field name
+ description: The header field name.
+ This will be canonicalized upon output,
+ so case-variant names will be understood
+ as the same header.
type: string
value:
description: The header field value
@@ -3589,8 +3676,7 @@ spec:
type: integer
grpc:
description: GRPC specifies an action involving
- a GRPC port. This is a beta field and requires
- enabling GRPCContainerProbe feature gate.
+ a GRPC port.
properties:
port:
description: Port number of the gRPC service.
@@ -3624,7 +3710,10 @@ spec:
header to be used in HTTP probes
properties:
name:
- description: The header field name
+ description: The header field name.
+ This will be canonicalized upon output,
+ so case-variant names will be understood
+ as the same header.
type: string
value:
description: The header field value
@@ -3719,6 +3808,28 @@ spec:
format: int32
type: integer
type: object
+ resizePolicy:
+ description: Resources resize policy for the container.
+ items:
+ description: ContainerResizePolicy represents resource
+ resize policy for the container.
+ properties:
+ resourceName:
+ description: 'Name of the resource to which
+ this resource resize policy applies. Supported
+ values: cpu, memory.'
+ type: string
+ restartPolicy:
+ description: Restart policy to apply when specified
+ resource is resized. If not specified, it
+ defaults to NotRequired.
+ type: string
+ required:
+ - resourceName
+ - restartPolicy
+ type: object
+ type: array
+ x-kubernetes-list-type: atomic
resources:
description: Resources are not allowed for ephemeral
containers. Ephemeral containers use spare resources
@@ -3769,10 +3880,16 @@ spec:
of compute resources required. If Requests is
omitted for a container, it defaults to Limits
if that is explicitly specified, otherwise to
- an implementation-defined value. More info:
- https://kubernetes.io/docs/concepts/configuration/manage-resources-containers/'
+ an implementation-defined value. Requests cannot
+ exceed Limits. More info: https://kubernetes.io/docs/concepts/configuration/manage-resources-containers/'
type: object
type: object
+ restartPolicy:
+ description: Restart policy for the container to manage
+ the restart behavior of each container within a
+ pod. This may only be set for init containers. You
+ cannot set this field on ephemeral containers.
+ type: string
securityContext:
description: 'Optional: SecurityContext defines the
security options the ephemeral container should
@@ -3908,8 +4025,9 @@ spec:
be used. The profile must be preconfigured
on the node to work. Must be a descending
path, relative to the kubelet's configured
- seccomp profile location. Must only be set
- if type is "Localhost".
+ seccomp profile location. Must be set if
+ type is "Localhost". Must NOT be set for
+ any other type.
type: string
type:
description: "type indicates which kind of
@@ -3946,17 +4064,12 @@ spec:
hostProcess:
description: HostProcess determines if a container
should be run as a 'Host Process' container.
- This field is alpha-level and will only
- be honored by components that enable the
- WindowsHostProcessContainers feature flag.
- Setting this field without the feature flag
- will result in errors when validating the
- Pod. All of a Pod's containers must have
- the same effective HostProcess value (it
- is not allowed to have a mix of HostProcess
- containers and non-HostProcess containers). In
- addition, if HostProcess is true then HostNetwork
- must also be set to true.
+ All of a Pod's containers must have the
+ same effective HostProcess value (it is
+ not allowed to have a mix of HostProcess
+ containers and non-HostProcess containers).
+ In addition, if HostProcess is true then
+ HostNetwork must also be set to true.
type: boolean
runAsUserName:
description: The UserName in Windows to run
@@ -3999,8 +4112,7 @@ spec:
type: integer
grpc:
description: GRPC specifies an action involving
- a GRPC port. This is a beta field and requires
- enabling GRPCContainerProbe feature gate.
+ a GRPC port.
properties:
port:
description: Port number of the gRPC service.
@@ -4034,7 +4146,10 @@ spec:
header to be used in HTTP probes
properties:
name:
- description: The header field name
+ description: The header field name.
+ This will be canonicalized upon output,
+ so case-variant names will be understood
+ as the same header.
type: string
value:
description: The header field value
@@ -4331,6 +4446,7 @@ spec:
uid?'
type: string
type: object
+ x-kubernetes-map-type: atomic
type: array
initContainers:
description: 'List of initialization containers belonging
@@ -4430,6 +4546,7 @@ spec:
required:
- key
type: object
+ x-kubernetes-map-type: atomic
fieldRef:
description: 'Selects a field of the pod:
supports metadata.name, metadata.namespace,
@@ -4449,6 +4566,7 @@ spec:
required:
- fieldPath
type: object
+ x-kubernetes-map-type: atomic
resourceFieldRef:
description: 'Selects a resource of the
container: only resources limits and requests
@@ -4476,6 +4594,7 @@ spec:
required:
- resource
type: object
+ x-kubernetes-map-type: atomic
secretKeyRef:
description: Selects a key of a secret in
the pod's namespace
@@ -4498,6 +4617,7 @@ spec:
required:
- key
type: object
+ x-kubernetes-map-type: atomic
type: object
required:
- name
@@ -4530,6 +4650,7 @@ spec:
must be defined
type: boolean
type: object
+ x-kubernetes-map-type: atomic
prefix:
description: An optional identifier to prepend
to each key in the ConfigMap. Must be a C_IDENTIFIER.
@@ -4548,6 +4669,7 @@ spec:
must be defined
type: boolean
type: object
+ x-kubernetes-map-type: atomic
type: object
type: array
image:
@@ -4612,7 +4734,11 @@ spec:
custom header to be used in HTTP probes
properties:
name:
- description: The header field name
+ description: The header field name.
+ This will be canonicalized upon
+ output, so case-variant names
+ will be understood as the same
+ header.
type: string
value:
description: The header field value
@@ -4720,7 +4846,11 @@ spec:
custom header to be used in HTTP probes
properties:
name:
- description: The header field name
+ description: The header field name.
+ This will be canonicalized upon
+ output, so case-variant names
+ will be understood as the same
+ header.
type: string
value:
description: The header field value
@@ -4807,8 +4937,7 @@ spec:
type: integer
grpc:
description: GRPC specifies an action involving
- a GRPC port. This is a beta field and requires
- enabling GRPCContainerProbe feature gate.
+ a GRPC port.
properties:
port:
description: Port number of the gRPC service.
@@ -4842,7 +4971,10 @@ spec:
header to be used in HTTP probes
properties:
name:
- description: The header field name
+ description: The header field name.
+ This will be canonicalized upon output,
+ so case-variant names will be understood
+ as the same header.
type: string
value:
description: The header field value
@@ -5024,8 +5156,7 @@ spec:
type: integer
grpc:
description: GRPC specifies an action involving
- a GRPC port. This is a beta field and requires
- enabling GRPCContainerProbe feature gate.
+ a GRPC port.
properties:
port:
description: Port number of the gRPC service.
@@ -5059,7 +5190,10 @@ spec:
header to be used in HTTP probes
properties:
name:
- description: The header field name
+ description: The header field name.
+ This will be canonicalized upon output,
+ so case-variant names will be understood
+ as the same header.
type: string
value:
description: The header field value
@@ -5154,6 +5288,28 @@ spec:
format: int32
type: integer
type: object
+ resizePolicy:
+ description: Resources resize policy for the container.
+ items:
+ description: ContainerResizePolicy represents resource
+ resize policy for the container.
+ properties:
+ resourceName:
+ description: 'Name of the resource to which
+ this resource resize policy applies. Supported
+ values: cpu, memory.'
+ type: string
+ restartPolicy:
+ description: Restart policy to apply when specified
+ resource is resized. If not specified, it
+ defaults to NotRequired.
+ type: string
+ required:
+ - resourceName
+ - restartPolicy
+ type: object
+ type: array
+ x-kubernetes-list-type: atomic
resources:
description: 'Compute Resources required by this container.
Cannot be updated. More info: https://kubernetes.io/docs/concepts/configuration/manage-resources-containers/'
@@ -5203,10 +5359,33 @@ spec:
of compute resources required. If Requests is
omitted for a container, it defaults to Limits
if that is explicitly specified, otherwise to
- an implementation-defined value. More info:
- https://kubernetes.io/docs/concepts/configuration/manage-resources-containers/'
+ an implementation-defined value. Requests cannot
+ exceed Limits. More info: https://kubernetes.io/docs/concepts/configuration/manage-resources-containers/'
type: object
type: object
+ restartPolicy:
+ description: 'RestartPolicy defines the restart behavior
+ of individual containers in a pod. This field may
+ only be set for init containers, and the only allowed
+ value is "Always". For non-init containers or when
+ this field is not specified, the restart behavior
+ is defined by the Pod''s restart policy and the
+ container type. Setting the RestartPolicy as "Always"
+ for the init container will have the following effect:
+ this init container will be continually restarted
+ on exit until all regular containers have terminated.
+ Once all regular containers have completed, all
+ init containers with restartPolicy "Always" will
+ be shut down. This lifecycle differs from normal
+ init containers and is often referred to as a "sidecar"
+ container. Although this init container still starts
+ in the init container sequence, it does not wait
+ for the container to complete before proceeding
+ to the next init container. Instead, the next init
+ container starts immediately after this init container
+ is started, or after any startupProbe has successfully
+ completed.'
+ type: string
securityContext:
description: 'SecurityContext defines the security
options the container should be run with. If set,
@@ -5342,8 +5521,9 @@ spec:
be used. The profile must be preconfigured
on the node to work. Must be a descending
path, relative to the kubelet's configured
- seccomp profile location. Must only be set
- if type is "Localhost".
+ seccomp profile location. Must be set if
+ type is "Localhost". Must NOT be set for
+ any other type.
type: string
type:
description: "type indicates which kind of
@@ -5380,17 +5560,12 @@ spec:
hostProcess:
description: HostProcess determines if a container
should be run as a 'Host Process' container.
- This field is alpha-level and will only
- be honored by components that enable the
- WindowsHostProcessContainers feature flag.
- Setting this field without the feature flag
- will result in errors when validating the
- Pod. All of a Pod's containers must have
- the same effective HostProcess value (it
- is not allowed to have a mix of HostProcess
- containers and non-HostProcess containers). In
- addition, if HostProcess is true then HostNetwork
- must also be set to true.
+ All of a Pod's containers must have the
+ same effective HostProcess value (it is
+ not allowed to have a mix of HostProcess
+ containers and non-HostProcess containers).
+ In addition, if HostProcess is true then
+ HostNetwork must also be set to true.
type: boolean
runAsUserName:
description: The UserName in Windows to run
@@ -5441,8 +5616,7 @@ spec:
type: integer
grpc:
description: GRPC specifies an action involving
- a GRPC port. This is a beta field and requires
- enabling GRPCContainerProbe feature gate.
+ a GRPC port.
properties:
port:
description: Port number of the gRPC service.
@@ -5476,7 +5650,10 @@ spec:
header to be used in HTTP probes
properties:
name:
- description: The header field name
+ description: The header field name.
+ This will be canonicalized upon output,
+ so case-variant names will be understood
+ as the same header.
type: string
value:
description: The header field value
@@ -5832,18 +6009,13 @@ spec:
will be used to create a new ResourceClaim,
which will be bound to this pod. When this pod
is deleted, the ResourceClaim will also be deleted.
- The name of the ResourceClaim will be -, where is the PodResourceClaim.Name.
- Pod validation will reject the pod if the concatenated
- name is not valid for a ResourceClaim (e.g.
- too long). \n An existing ResourceClaim with
- that name that is not owned by the pod will
- not be used for the pod to avoid using an unrelated
- resource by mistake. Scheduling and pod startup
- are then blocked until the unrelated ResourceClaim
- is removed. \n This field is immutable and no
- changes will be made to the corresponding ResourceClaim
- by the control plane after creating the ResourceClaim."
+ The pod name and resource name, along with a
+ generated component, will be used to form a
+ unique name for the ResourceClaim, which will
+ be recorded in pod.status.resourceClaimStatuses.
+ \n This field is immutable and no changes will
+ be made to the corresponding ResourceClaim by
+ the control plane after creating the ResourceClaim."
type: string
type: object
required:
@@ -5855,8 +6027,9 @@ spec:
x-kubernetes-list-type: map
restartPolicy:
description: 'Restart policy for all containers within the
- pod. One of Always, OnFailure, Never. Default to Always.
- More info: https://kubernetes.io/docs/concepts/workloads/pods/pod-lifecycle/#restart-policy'
+ pod. One of Always, OnFailure, Never. In some contexts,
+ only a subset of those values may be permitted. Default
+ to Always. More info: https://kubernetes.io/docs/concepts/workloads/pods/pod-lifecycle/#restart-policy'
type: string
runtimeClassName:
description: 'RuntimeClassName refers to a RuntimeClass
@@ -5874,10 +6047,13 @@ spec:
type: string
schedulingGates:
description: "SchedulingGates is an opaque list of values
- that if specified will block scheduling the pod. More
- info: https://git.k8s.io/enhancements/keps/sig-scheduling/3521-pod-scheduling-readiness.
- \n This is an alpha-level feature enabled by PodSchedulingReadiness
- feature gate."
+ that if specified will block scheduling the pod. If schedulingGates
+ is not empty, the pod will stay in the SchedulingGated
+ state and the scheduler will not attempt to schedule the
+ pod. \n SchedulingGates can only be set at pod creation
+ time, and be removed only afterwards. \n This is a beta
+ feature enabled by the PodSchedulingReadiness feature
+ gate."
items:
description: PodSchedulingGate is associated to a Pod
to guard its scheduling.
@@ -5988,7 +6164,8 @@ spec:
The profile must be preconfigured on the node
to work. Must be a descending path, relative to
the kubelet's configured seccomp profile location.
- Must only be set if type is "Localhost".
+ Must be set if type is "Localhost". Must NOT be
+ set for any other type.
type: string
type:
description: "type indicates which kind of seccomp
@@ -6058,15 +6235,11 @@ spec:
type: string
hostProcess:
description: HostProcess determines if a container
- should be run as a 'Host Process' container. This
- field is alpha-level and will only be honored
- by components that enable the WindowsHostProcessContainers
- feature flag. Setting this field without the feature
- flag will result in errors when validating the
- Pod. All of a Pod's containers must have the same
- effective HostProcess value (it is not allowed
- to have a mix of HostProcess containers and non-HostProcess
- containers). In addition, if HostProcess is true
+ should be run as a 'Host Process' container. All
+ of a Pod's containers must have the same effective
+ HostProcess value (it is not allowed to have a
+ mix of HostProcess containers and non-HostProcess
+ containers). In addition, if HostProcess is true
then HostNetwork must also be set to true.
type: boolean
runAsUserName:
@@ -6228,16 +6401,22 @@ spec:
The requirements are ANDed.
type: object
type: object
+ x-kubernetes-map-type: atomic
matchLabelKeys:
- description: MatchLabelKeys is a set of pod label
+ description: "MatchLabelKeys is a set of pod label
keys to select the pods over which spreading will
be calculated. The keys are used to lookup values
from the incoming pod labels, those key-value labels
are ANDed with labelSelector to select the group
of existing pods over which spreading will be calculated
- for the incoming pod. Keys that don't exist in the
- incoming pod labels will be ignored. A null or empty
- list means only match against labelSelector.
+ for the incoming pod. The same key is forbidden
+ to exist in both MatchLabelKeys and LabelSelector.
+ MatchLabelKeys cannot be set when LabelSelector
+ isn't set. Keys that don't exist in the incoming
+ pod labels will be ignored. A null or empty list
+ means only match against labelSelector. \n This
+ is a beta field and requires the MatchLabelKeysInPodTopologySpread
+ feature gate to be enabled (enabled by default)."
items:
type: string
type: array
@@ -6506,6 +6685,7 @@ spec:
kind, uid?'
type: string
type: object
+ x-kubernetes-map-type: atomic
user:
description: 'user is optional: User is the rados
user name, default is admin More info: https://examples.k8s.io/volumes/cephfs/README.md#how-to-use-it'
@@ -6542,6 +6722,7 @@ spec:
kind, uid?'
type: string
type: object
+ x-kubernetes-map-type: atomic
volumeID:
description: 'volumeID used to identify the volume
in cinder. More info: https://examples.k8s.io/mysql-cinder-pd/README.md'
@@ -6623,6 +6804,7 @@ spec:
or its keys must be defined
type: boolean
type: object
+ x-kubernetes-map-type: atomic
csi:
description: csi (Container Storage Interface) represents
ephemeral storage that is handled by certain external
@@ -6657,6 +6839,7 @@ spec:
kind, uid?'
type: string
type: object
+ x-kubernetes-map-type: atomic
readOnly:
description: readOnly specifies a read-only configuration
for the volume. Defaults to false (read/write).
@@ -6716,6 +6899,7 @@ spec:
required:
- fieldPath
type: object
+ x-kubernetes-map-type: atomic
mode:
description: 'Optional: mode bits used to
set permissions on this file, must be
@@ -6764,6 +6948,7 @@ spec:
required:
- resource
type: object
+ x-kubernetes-map-type: atomic
required:
- path
type: object
@@ -6792,7 +6977,7 @@ spec:
specified here and the sum of memory limits
of all containers in a pod. The default is nil
which means that the limit is undefined. More
- info: http://kubernetes.io/docs/user-guide/volumes#emptydir'
+ info: https://kubernetes.io/docs/concepts/storage/volumes#emptydir'
pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$
x-kubernetes-int-or-string: true
type: object
@@ -6918,6 +7103,7 @@ spec:
- kind
- name
type: object
+ x-kubernetes-map-type: atomic
dataSourceRef:
description: 'dataSourceRef specifies
the object from which to populate the
@@ -7052,7 +7238,8 @@ spec:
for a container, it defaults to
Limits if that is explicitly specified,
otherwise to an implementation-defined
- value. More info: https://kubernetes.io/docs/concepts/configuration/manage-resources-containers/'
+ value. Requests cannot exceed Limits.
+ More info: https://kubernetes.io/docs/concepts/configuration/manage-resources-containers/'
type: object
type: object
selector:
@@ -7112,6 +7299,7 @@ spec:
ANDed.
type: object
type: object
+ x-kubernetes-map-type: atomic
storageClassName:
description: 'storageClassName is the
name of the StorageClass required by
@@ -7213,6 +7401,7 @@ spec:
kind, uid?'
type: string
type: object
+ x-kubernetes-map-type: atomic
required:
- driver
type: object
@@ -7406,6 +7595,7 @@ spec:
kind, uid?'
type: string
type: object
+ x-kubernetes-map-type: atomic
targetPortal:
description: targetPortal is iSCSI Target Portal.
The Portal is either an IP or ip_addr:port if
@@ -7593,6 +7783,7 @@ spec:
defined
type: boolean
type: object
+ x-kubernetes-map-type: atomic
downwardAPI:
description: downwardAPI information about
the downwardAPI data to project
@@ -7625,6 +7816,7 @@ spec:
required:
- fieldPath
type: object
+ x-kubernetes-map-type: atomic
mode:
description: 'Optional: mode bits
used to set permissions on this
@@ -7680,6 +7872,7 @@ spec:
required:
- resource
type: object
+ x-kubernetes-map-type: atomic
required:
- path
type: object
@@ -7755,6 +7948,7 @@ spec:
be defined
type: boolean
type: object
+ x-kubernetes-map-type: atomic
serviceAccountToken:
description: serviceAccountToken is information
about the serviceAccountToken data to
@@ -7882,6 +8076,7 @@ spec:
kind, uid?'
type: string
type: object
+ x-kubernetes-map-type: atomic
user:
description: 'user is the rados user name. Default
is admin. More info: https://examples.k8s.io/volumes/rbd/README.md#how-to-use-it'
@@ -7927,6 +8122,7 @@ spec:
kind, uid?'
type: string
type: object
+ x-kubernetes-map-type: atomic
sslEnabled:
description: sslEnabled Flag enable/disable SSL
communication with Gateway, default false
@@ -8054,6 +8250,7 @@ spec:
kind, uid?'
type: string
type: object
+ x-kubernetes-map-type: atomic
volumeName:
description: volumeName is the human-readable
name of the StorageOS volume. Volume names
@@ -8318,6 +8515,235 @@ spec:
- Parallel
type: string
type: object
+ rsmSpec:
+ description: RSMSpec defines workload related spec of this component.
+ start from KB 0.7.0, RSM(ReplicatedStateMachineSpec) will
+ be the underlying CR which powers all kinds of workload in
+ KB. RSM is an enhanced stateful workload extension dedicated
+ for heavy-state workloads like databases.
+ properties:
+ memberUpdateStrategy:
+ description: 'MemberUpdateStrategy, Members(Pods) update
+ strategy. serial: update Members one by one that guarantee
+ minimum component unavailable time. Learner -> Follower(with
+ AccessMode=none) -> Follower(with AccessMode=readonly)
+ -> Follower(with AccessMode=readWrite) -> Leader bestEffortParallel:
+ update Members in parallel that guarantee minimum component
+ un-writable time. Learner, Follower(minority) in parallel
+ -> Follower(majority) -> Leader, keep majority online
+ all the time. parallel: force parallel'
+ enum:
+ - Serial
+ - BestEffortParallel
+ - Parallel
+ type: string
+ membershipReconfiguration:
+ description: MembershipReconfiguration provides actions
+ to do membership dynamic reconfiguration.
+ properties:
+ logSyncAction:
+ description: LogSyncAction specifies how to trigger
+ the new member to start log syncing previous none-nil
+ action's Image wil be used if not configured
+ properties:
+ command:
+ description: Command will be executed in Container
+ to retrieve or process role info
+ items:
+ type: string
+ type: array
+ image:
+ description: utility image contains command that
+ can be used to retrieve of process role info
+ type: string
+ required:
+ - command
+ type: object
+ memberJoinAction:
+ description: MemberJoinAction specifies how to add member
+ previous none-nil action's Image wil be used if not
+ configured
+ properties:
+ command:
+ description: Command will be executed in Container
+ to retrieve or process role info
+ items:
+ type: string
+ type: array
+ image:
+ description: utility image contains command that
+ can be used to retrieve of process role info
+ type: string
+ required:
+ - command
+ type: object
+ memberLeaveAction:
+ description: MemberLeaveAction specifies how to remove
+ member previous none-nil action's Image wil be used
+ if not configured
+ properties:
+ command:
+ description: Command will be executed in Container
+ to retrieve or process role info
+ items:
+ type: string
+ type: array
+ image:
+ description: utility image contains command that
+ can be used to retrieve of process role info
+ type: string
+ required:
+ - command
+ type: object
+ promoteAction:
+ description: PromoteAction specifies how to tell the
+ cluster that the new member can join voting now previous
+ none-nil action's Image wil be used if not configured
+ properties:
+ command:
+ description: Command will be executed in Container
+ to retrieve or process role info
+ items:
+ type: string
+ type: array
+ image:
+ description: utility image contains command that
+ can be used to retrieve of process role info
+ type: string
+ required:
+ - command
+ type: object
+ switchoverAction:
+ description: SwitchoverAction specifies how to do switchover
+ latest [BusyBox](https://busybox.net/) image will
+ be used if Image not configured
+ properties:
+ command:
+ description: Command will be executed in Container
+ to retrieve or process role info
+ items:
+ type: string
+ type: array
+ image:
+ description: utility image contains command that
+ can be used to retrieve of process role info
+ type: string
+ required:
+ - command
+ type: object
+ type: object
+ roleProbe:
+ description: RoleProbe provides method to probe role.
+ properties:
+ failureThreshold:
+ default: 3
+ description: Minimum consecutive failures for the probe
+ to be considered failed after having succeeded. Defaults
+ to 3. Minimum value is 1.
+ format: int32
+ minimum: 1
+ type: integer
+ initialDelaySeconds:
+ default: 0
+ description: Number of seconds after the container has
+ started before role probe has started.
+ format: int32
+ minimum: 0
+ type: integer
+ periodSeconds:
+ default: 2
+ description: How often (in seconds) to perform the probe.
+ Default to 2 seconds. Minimum value is 1.
+ format: int32
+ minimum: 1
+ type: integer
+ probeActions:
+ description: 'ProbeActions define Actions to be taken
+ in serial. after all actions done, the final output
+ should be a single string of the role name defined
+ in spec.Roles latest [BusyBox](https://busybox.net/)
+ image will be used if Image not configured Environment
+ variables can be used in Command: - v_KB_RSM_LAST_STDOUT
+ stdout from last action, watch ''v_'' prefixed - KB_RSM_USERNAME
+ username part of credential - KB_RSM_PASSWORD password
+ part of credential'
+ items:
+ properties:
+ command:
+ description: Command will be executed in Container
+ to retrieve or process role info
+ items:
+ type: string
+ type: array
+ image:
+ description: utility image contains command that
+ can be used to retrieve of process role info
+ type: string
+ required:
+ - command
+ type: object
+ type: array
+ roleUpdateMechanism:
+ default: None
+ description: RoleUpdateMechanism specifies the way how
+ pod role label being updated.
+ enum:
+ - ReadinessProbeEventUpdate
+ - DirectAPIServerEventUpdate
+ - None
+ type: string
+ successThreshold:
+ default: 1
+ description: Minimum consecutive successes for the probe
+ to be considered successful after having failed. Defaults
+ to 1. Minimum value is 1.
+ format: int32
+ minimum: 1
+ type: integer
+ timeoutSeconds:
+ default: 1
+ description: Number of seconds after which the probe
+ times out. Defaults to 1 second. Minimum value is
+ 1.
+ format: int32
+ minimum: 1
+ type: integer
+ required:
+ - probeActions
+ type: object
+ roles:
+ description: Roles, a list of roles defined in the system.
+ items:
+ properties:
+ accessMode:
+ default: ReadWrite
+ description: AccessMode, what service this member
+ capable.
+ enum:
+ - None
+ - Readonly
+ - ReadWrite
+ type: string
+ canVote:
+ default: true
+ description: CanVote, whether this member has voting
+ rights
+ type: boolean
+ isLeader:
+ default: false
+ description: IsLeader, whether this member is the
+ leader
+ type: boolean
+ name:
+ default: leader
+ description: Name, role name.
+ type: string
+ required:
+ - accessMode
+ - name
+ type: object
+ type: array
+ type: object
scriptSpecs:
description: The scriptSpec field provided by provider, and
finally this configTemplateRefs will be rendered into the
@@ -8698,6 +9124,7 @@ spec:
required:
- key
type: object
+ x-kubernetes-map-type: atomic
fieldRef:
description: 'Selects a field of the pod:
supports metadata.name, metadata.namespace,
@@ -8717,6 +9144,7 @@ spec:
required:
- fieldPath
type: object
+ x-kubernetes-map-type: atomic
resourceFieldRef:
description: 'Selects a resource of the
container: only resources limits and
@@ -8745,6 +9173,7 @@ spec:
required:
- resource
type: object
+ x-kubernetes-map-type: atomic
secretKeyRef:
description: Selects a key of a secret
in the pod's namespace
@@ -8767,6 +9196,7 @@ spec:
required:
- key
type: object
+ x-kubernetes-map-type: atomic
type: object
required:
- name
@@ -8867,6 +9297,7 @@ spec:
required:
- key
type: object
+ x-kubernetes-map-type: atomic
fieldRef:
description: 'Selects a field of the pod:
supports metadata.name, metadata.namespace,
@@ -8886,6 +9317,7 @@ spec:
required:
- fieldPath
type: object
+ x-kubernetes-map-type: atomic
resourceFieldRef:
description: 'Selects a resource of the
container: only resources limits and
@@ -8914,6 +9346,7 @@ spec:
required:
- resource
type: object
+ x-kubernetes-map-type: atomic
secretKeyRef:
description: Selects a key of a secret
in the pod's namespace
@@ -8936,6 +9369,7 @@ spec:
required:
- key
type: object
+ x-kubernetes-map-type: atomic
type: object
required:
- name
@@ -9118,6 +9552,7 @@ spec:
required:
- key
type: object
+ x-kubernetes-map-type: atomic
fieldRef:
description: 'Selects a field of the pod:
supports metadata.name, metadata.namespace,
@@ -9137,6 +9572,7 @@ spec:
required:
- fieldPath
type: object
+ x-kubernetes-map-type: atomic
resourceFieldRef:
description: 'Selects a resource of the container:
only resources limits and requests (limits.cpu,
@@ -9163,6 +9599,7 @@ spec:
required:
- resource
type: object
+ x-kubernetes-map-type: atomic
secretKeyRef:
description: Selects a key of a secret in
the pod's namespace
@@ -9185,6 +9622,7 @@ spec:
required:
- key
type: object
+ x-kubernetes-map-type: atomic
type: object
required:
- name
@@ -9323,10 +9761,11 @@ spec:
- workloadType
type: object
x-kubernetes-validations:
- - message: componentDefs.consensusSpec is required when componentDefs.workloadType
- is Consensus, and forbidden otherwise
+ - message: componentDefs.consensusSpec(deprecated) or componentDefs.rsmSpec(recommended)
+ is required when componentDefs.workloadType is Consensus, and
+ forbidden otherwise
rule: 'has(self.workloadType) && self.workloadType == ''Consensus''
- ? has(self.consensusSpec) : !has(self.consensusSpec)'
+ ? (has(self.consensusSpec) || has(self.rsmSpec)) : !has(self.consensusSpec)'
minItems: 1
type: array
x-kubernetes-list-map-keys:
diff --git a/deploy/helm/crds/apps.kubeblocks.io_clusters.yaml b/deploy/helm/crds/apps.kubeblocks.io_clusters.yaml
index 18c2abf550a..d741c62fef4 100644
--- a/deploy/helm/crds/apps.kubeblocks.io_clusters.yaml
+++ b/deploy/helm/crds/apps.kubeblocks.io_clusters.yaml
@@ -2,8 +2,7 @@ apiVersion: apiextensions.k8s.io/v1
kind: CustomResourceDefinition
metadata:
annotations:
- controller-gen.kubebuilder.io/version: v0.9.0
- creationTimestamp: null
+ controller-gen.kubebuilder.io/version: v0.12.1
labels:
app.kubernetes.io/name: kubeblocks
name: clusters.apps.kubeblocks.io
@@ -119,15 +118,7 @@ spec:
description: enabled defines whether to enable automated backup.
type: boolean
method:
- allOf:
- - enum:
- - snapshot
- - backupTool
- - enum:
- - snapshot
- - backupTool
- default: snapshot
- description: 'backup method, support: snapshot, backupTool.'
+ description: backup method name to use, that is defined in backupPolicy.
type: string
pitrEnabled:
default: false
@@ -139,11 +130,14 @@ spec:
will use the default backupRepo.
type: string
retentionPeriod:
- default: 1d
- description: retentionPeriod is a time string ending with the
- 'd'|'D'|'h'|'H' character to describe how long the Backup should
- be retained. if not set, will be retained forever.
- pattern: ^\d+[d|D|h|H]$
+ default: 7d
+ description: "retentionPeriod determines a duration up to which
+ the backup should be kept. controller will remove all backups
+ that are older than the RetentionPeriod. For example, RetentionPeriod
+ of `30d` will keep only the backups of last 30 days. Sample
+ duration format: - years: \t2y - months: \t6mo - days: \t\t30d
+ - hours: \t12h - minutes: \t30m You can also combine the above
+ durations. For example: 30d12h30m"
type: string
startingDeadlineMinutes:
description: startingDeadlineMinutes defines the deadline in minutes
@@ -153,8 +147,6 @@ spec:
maximum: 1440
minimum: 0
type: integer
- required:
- - method
type: object
clusterDefinitionRef:
description: Cluster referencing ClusterDefinition name. This is an
@@ -362,8 +354,8 @@ spec:
description: 'Requests describes the minimum amount of compute
resources required. If Requests is omitted for a container,
it defaults to Limits if that is explicitly specified,
- otherwise to an implementation-defined value. More info:
- https://kubernetes.io/docs/concepts/configuration/manage-resources-containers/'
+ otherwise to an implementation-defined value. Requests
+ cannot exceed Limits. More info: https://kubernetes.io/docs/concepts/configuration/manage-resources-containers/'
type: object
type: object
x-kubernetes-preserve-unknown-fields: true
@@ -622,8 +614,8 @@ spec:
of compute resources required. If Requests is
omitted for a container, it defaults to Limits
if that is explicitly specified, otherwise to
- an implementation-defined value. More info:
- https://kubernetes.io/docs/concepts/configuration/manage-resources-containers/'
+ an implementation-defined value. Requests cannot
+ exceed Limits. More info: https://kubernetes.io/docs/concepts/configuration/manage-resources-containers/'
type: object
type: object
x-kubernetes-preserve-unknown-fields: true
@@ -873,6 +865,48 @@ spec:
required:
- leader
type: object
+ membersStatus:
+ description: members' status.
+ items:
+ properties:
+ podName:
+ default: Unknown
+ description: PodName pod name.
+ type: string
+ role:
+ properties:
+ accessMode:
+ default: ReadWrite
+ description: AccessMode, what service this member
+ capable.
+ enum:
+ - None
+ - Readonly
+ - ReadWrite
+ type: string
+ canVote:
+ default: true
+ description: CanVote, whether this member has voting
+ rights
+ type: boolean
+ isLeader:
+ default: false
+ description: IsLeader, whether this member is the
+ leader
+ type: boolean
+ name:
+ default: leader
+ description: Name, role name.
+ type: string
+ required:
+ - accessMode
+ - name
+ type: object
+ required:
+ - podName
+ - role
+ type: object
+ type: array
message:
additionalProperties:
type: string
diff --git a/deploy/helm/crds/apps.kubeblocks.io_clusterversions.yaml b/deploy/helm/crds/apps.kubeblocks.io_clusterversions.yaml
index 77302045da9..3e9507ef482 100644
--- a/deploy/helm/crds/apps.kubeblocks.io_clusterversions.yaml
+++ b/deploy/helm/crds/apps.kubeblocks.io_clusterversions.yaml
@@ -2,8 +2,7 @@ apiVersion: apiextensions.k8s.io/v1
kind: CustomResourceDefinition
metadata:
annotations:
- controller-gen.kubebuilder.io/version: v0.9.0
- creationTimestamp: null
+ controller-gen.kubebuilder.io/version: v0.12.1
labels:
app.kubernetes.io/name: kubeblocks
name: clusterversions.apps.kubeblocks.io
@@ -230,6 +229,7 @@ spec:
required:
- key
type: object
+ x-kubernetes-map-type: atomic
fieldRef:
description: 'Selects a field of the pod:
supports metadata.name, metadata.namespace,
@@ -249,6 +249,7 @@ spec:
required:
- fieldPath
type: object
+ x-kubernetes-map-type: atomic
resourceFieldRef:
description: 'Selects a resource of the container:
only resources limits and requests (limits.cpu,
@@ -275,6 +276,7 @@ spec:
required:
- resource
type: object
+ x-kubernetes-map-type: atomic
secretKeyRef:
description: Selects a key of a secret in
the pod's namespace
@@ -297,6 +299,7 @@ spec:
required:
- key
type: object
+ x-kubernetes-map-type: atomic
type: object
required:
- name
@@ -370,6 +373,7 @@ spec:
required:
- key
type: object
+ x-kubernetes-map-type: atomic
fieldRef:
description: 'Selects a field of the pod:
supports metadata.name, metadata.namespace,
@@ -389,6 +393,7 @@ spec:
required:
- fieldPath
type: object
+ x-kubernetes-map-type: atomic
resourceFieldRef:
description: 'Selects a resource of the container:
only resources limits and requests (limits.cpu,
@@ -415,6 +420,7 @@ spec:
required:
- resource
type: object
+ x-kubernetes-map-type: atomic
secretKeyRef:
description: Selects a key of a secret in
the pod's namespace
@@ -437,6 +443,7 @@ spec:
required:
- key
type: object
+ x-kubernetes-map-type: atomic
type: object
required:
- name
@@ -545,6 +552,7 @@ spec:
required:
- key
type: object
+ x-kubernetes-map-type: atomic
fieldRef:
description: 'Selects a field of the pod:
supports metadata.name, metadata.namespace,
@@ -564,6 +572,7 @@ spec:
required:
- fieldPath
type: object
+ x-kubernetes-map-type: atomic
resourceFieldRef:
description: 'Selects a resource of the
container: only resources limits and requests
@@ -591,6 +600,7 @@ spec:
required:
- resource
type: object
+ x-kubernetes-map-type: atomic
secretKeyRef:
description: Selects a key of a secret in
the pod's namespace
@@ -613,6 +623,7 @@ spec:
required:
- key
type: object
+ x-kubernetes-map-type: atomic
type: object
required:
- name
@@ -645,6 +656,7 @@ spec:
must be defined
type: boolean
type: object
+ x-kubernetes-map-type: atomic
prefix:
description: An optional identifier to prepend
to each key in the ConfigMap. Must be a C_IDENTIFIER.
@@ -663,6 +675,7 @@ spec:
must be defined
type: boolean
type: object
+ x-kubernetes-map-type: atomic
type: object
type: array
image:
@@ -727,7 +740,11 @@ spec:
custom header to be used in HTTP probes
properties:
name:
- description: The header field name
+ description: The header field name.
+ This will be canonicalized upon
+ output, so case-variant names
+ will be understood as the same
+ header.
type: string
value:
description: The header field value
@@ -835,7 +852,11 @@ spec:
custom header to be used in HTTP probes
properties:
name:
- description: The header field name
+ description: The header field name.
+ This will be canonicalized upon
+ output, so case-variant names
+ will be understood as the same
+ header.
type: string
value:
description: The header field value
@@ -922,8 +943,7 @@ spec:
type: integer
grpc:
description: GRPC specifies an action involving
- a GRPC port. This is a beta field and requires
- enabling GRPCContainerProbe feature gate.
+ a GRPC port.
properties:
port:
description: Port number of the gRPC service.
@@ -957,7 +977,10 @@ spec:
header to be used in HTTP probes
properties:
name:
- description: The header field name
+ description: The header field name.
+ This will be canonicalized upon output,
+ so case-variant names will be understood
+ as the same header.
type: string
value:
description: The header field value
@@ -1139,8 +1162,7 @@ spec:
type: integer
grpc:
description: GRPC specifies an action involving
- a GRPC port. This is a beta field and requires
- enabling GRPCContainerProbe feature gate.
+ a GRPC port.
properties:
port:
description: Port number of the gRPC service.
@@ -1174,7 +1196,10 @@ spec:
header to be used in HTTP probes
properties:
name:
- description: The header field name
+ description: The header field name.
+ This will be canonicalized upon output,
+ so case-variant names will be understood
+ as the same header.
type: string
value:
description: The header field value
@@ -1269,6 +1294,28 @@ spec:
format: int32
type: integer
type: object
+ resizePolicy:
+ description: Resources resize policy for the container.
+ items:
+ description: ContainerResizePolicy represents resource
+ resize policy for the container.
+ properties:
+ resourceName:
+ description: 'Name of the resource to which
+ this resource resize policy applies. Supported
+ values: cpu, memory.'
+ type: string
+ restartPolicy:
+ description: Restart policy to apply when specified
+ resource is resized. If not specified, it
+ defaults to NotRequired.
+ type: string
+ required:
+ - resourceName
+ - restartPolicy
+ type: object
+ type: array
+ x-kubernetes-list-type: atomic
resources:
description: 'Compute Resources required by this container.
Cannot be updated. More info: https://kubernetes.io/docs/concepts/configuration/manage-resources-containers/'
@@ -1318,10 +1365,33 @@ spec:
of compute resources required. If Requests is
omitted for a container, it defaults to Limits
if that is explicitly specified, otherwise to
- an implementation-defined value. More info:
- https://kubernetes.io/docs/concepts/configuration/manage-resources-containers/'
+ an implementation-defined value. Requests cannot
+ exceed Limits. More info: https://kubernetes.io/docs/concepts/configuration/manage-resources-containers/'
type: object
type: object
+ restartPolicy:
+ description: 'RestartPolicy defines the restart behavior
+ of individual containers in a pod. This field may
+ only be set for init containers, and the only allowed
+ value is "Always". For non-init containers or when
+ this field is not specified, the restart behavior
+ is defined by the Pod''s restart policy and the
+ container type. Setting the RestartPolicy as "Always"
+ for the init container will have the following effect:
+ this init container will be continually restarted
+ on exit until all regular containers have terminated.
+ Once all regular containers have completed, all
+ init containers with restartPolicy "Always" will
+ be shut down. This lifecycle differs from normal
+ init containers and is often referred to as a "sidecar"
+ container. Although this init container still starts
+ in the init container sequence, it does not wait
+ for the container to complete before proceeding
+ to the next init container. Instead, the next init
+ container starts immediately after this init container
+ is started, or after any startupProbe has successfully
+ completed.'
+ type: string
securityContext:
description: 'SecurityContext defines the security
options the container should be run with. If set,
@@ -1457,8 +1527,9 @@ spec:
be used. The profile must be preconfigured
on the node to work. Must be a descending
path, relative to the kubelet's configured
- seccomp profile location. Must only be set
- if type is "Localhost".
+ seccomp profile location. Must be set if
+ type is "Localhost". Must NOT be set for
+ any other type.
type: string
type:
description: "type indicates which kind of
@@ -1495,17 +1566,12 @@ spec:
hostProcess:
description: HostProcess determines if a container
should be run as a 'Host Process' container.
- This field is alpha-level and will only
- be honored by components that enable the
- WindowsHostProcessContainers feature flag.
- Setting this field without the feature flag
- will result in errors when validating the
- Pod. All of a Pod's containers must have
- the same effective HostProcess value (it
- is not allowed to have a mix of HostProcess
- containers and non-HostProcess containers). In
- addition, if HostProcess is true then HostNetwork
- must also be set to true.
+ All of a Pod's containers must have the
+ same effective HostProcess value (it is
+ not allowed to have a mix of HostProcess
+ containers and non-HostProcess containers).
+ In addition, if HostProcess is true then
+ HostNetwork must also be set to true.
type: boolean
runAsUserName:
description: The UserName in Windows to run
@@ -1556,8 +1622,7 @@ spec:
type: integer
grpc:
description: GRPC specifies an action involving
- a GRPC port. This is a beta field and requires
- enabling GRPCContainerProbe feature gate.
+ a GRPC port.
properties:
port:
description: Port number of the gRPC service.
@@ -1591,7 +1656,10 @@ spec:
header to be used in HTTP probes
properties:
name:
- description: The header field name
+ description: The header field name.
+ This will be canonicalized upon output,
+ so case-variant names will be understood
+ as the same header.
type: string
value:
description: The header field value
@@ -1900,6 +1968,7 @@ spec:
required:
- key
type: object
+ x-kubernetes-map-type: atomic
fieldRef:
description: 'Selects a field of the pod:
supports metadata.name, metadata.namespace,
@@ -1919,6 +1988,7 @@ spec:
required:
- fieldPath
type: object
+ x-kubernetes-map-type: atomic
resourceFieldRef:
description: 'Selects a resource of the
container: only resources limits and requests
@@ -1946,6 +2016,7 @@ spec:
required:
- resource
type: object
+ x-kubernetes-map-type: atomic
secretKeyRef:
description: Selects a key of a secret in
the pod's namespace
@@ -1968,6 +2039,7 @@ spec:
required:
- key
type: object
+ x-kubernetes-map-type: atomic
type: object
required:
- name
@@ -2000,6 +2072,7 @@ spec:
must be defined
type: boolean
type: object
+ x-kubernetes-map-type: atomic
prefix:
description: An optional identifier to prepend
to each key in the ConfigMap. Must be a C_IDENTIFIER.
@@ -2018,6 +2091,7 @@ spec:
must be defined
type: boolean
type: object
+ x-kubernetes-map-type: atomic
type: object
type: array
image:
@@ -2082,7 +2156,11 @@ spec:
custom header to be used in HTTP probes
properties:
name:
- description: The header field name
+ description: The header field name.
+ This will be canonicalized upon
+ output, so case-variant names
+ will be understood as the same
+ header.
type: string
value:
description: The header field value
@@ -2190,7 +2268,11 @@ spec:
custom header to be used in HTTP probes
properties:
name:
- description: The header field name
+ description: The header field name.
+ This will be canonicalized upon
+ output, so case-variant names
+ will be understood as the same
+ header.
type: string
value:
description: The header field value
@@ -2277,8 +2359,7 @@ spec:
type: integer
grpc:
description: GRPC specifies an action involving
- a GRPC port. This is a beta field and requires
- enabling GRPCContainerProbe feature gate.
+ a GRPC port.
properties:
port:
description: Port number of the gRPC service.
@@ -2312,7 +2393,10 @@ spec:
header to be used in HTTP probes
properties:
name:
- description: The header field name
+ description: The header field name.
+ This will be canonicalized upon output,
+ so case-variant names will be understood
+ as the same header.
type: string
value:
description: The header field value
@@ -2494,8 +2578,7 @@ spec:
type: integer
grpc:
description: GRPC specifies an action involving
- a GRPC port. This is a beta field and requires
- enabling GRPCContainerProbe feature gate.
+ a GRPC port.
properties:
port:
description: Port number of the gRPC service.
@@ -2529,7 +2612,10 @@ spec:
header to be used in HTTP probes
properties:
name:
- description: The header field name
+ description: The header field name.
+ This will be canonicalized upon output,
+ so case-variant names will be understood
+ as the same header.
type: string
value:
description: The header field value
@@ -2624,6 +2710,28 @@ spec:
format: int32
type: integer
type: object
+ resizePolicy:
+ description: Resources resize policy for the container.
+ items:
+ description: ContainerResizePolicy represents resource
+ resize policy for the container.
+ properties:
+ resourceName:
+ description: 'Name of the resource to which
+ this resource resize policy applies. Supported
+ values: cpu, memory.'
+ type: string
+ restartPolicy:
+ description: Restart policy to apply when specified
+ resource is resized. If not specified, it
+ defaults to NotRequired.
+ type: string
+ required:
+ - resourceName
+ - restartPolicy
+ type: object
+ type: array
+ x-kubernetes-list-type: atomic
resources:
description: 'Compute Resources required by this container.
Cannot be updated. More info: https://kubernetes.io/docs/concepts/configuration/manage-resources-containers/'
@@ -2673,10 +2781,33 @@ spec:
of compute resources required. If Requests is
omitted for a container, it defaults to Limits
if that is explicitly specified, otherwise to
- an implementation-defined value. More info:
- https://kubernetes.io/docs/concepts/configuration/manage-resources-containers/'
+ an implementation-defined value. Requests cannot
+ exceed Limits. More info: https://kubernetes.io/docs/concepts/configuration/manage-resources-containers/'
type: object
type: object
+ restartPolicy:
+ description: 'RestartPolicy defines the restart behavior
+ of individual containers in a pod. This field may
+ only be set for init containers, and the only allowed
+ value is "Always". For non-init containers or when
+ this field is not specified, the restart behavior
+ is defined by the Pod''s restart policy and the
+ container type. Setting the RestartPolicy as "Always"
+ for the init container will have the following effect:
+ this init container will be continually restarted
+ on exit until all regular containers have terminated.
+ Once all regular containers have completed, all
+ init containers with restartPolicy "Always" will
+ be shut down. This lifecycle differs from normal
+ init containers and is often referred to as a "sidecar"
+ container. Although this init container still starts
+ in the init container sequence, it does not wait
+ for the container to complete before proceeding
+ to the next init container. Instead, the next init
+ container starts immediately after this init container
+ is started, or after any startupProbe has successfully
+ completed.'
+ type: string
securityContext:
description: 'SecurityContext defines the security
options the container should be run with. If set,
@@ -2812,8 +2943,9 @@ spec:
be used. The profile must be preconfigured
on the node to work. Must be a descending
path, relative to the kubelet's configured
- seccomp profile location. Must only be set
- if type is "Localhost".
+ seccomp profile location. Must be set if
+ type is "Localhost". Must NOT be set for
+ any other type.
type: string
type:
description: "type indicates which kind of
@@ -2850,17 +2982,12 @@ spec:
hostProcess:
description: HostProcess determines if a container
should be run as a 'Host Process' container.
- This field is alpha-level and will only
- be honored by components that enable the
- WindowsHostProcessContainers feature flag.
- Setting this field without the feature flag
- will result in errors when validating the
- Pod. All of a Pod's containers must have
- the same effective HostProcess value (it
- is not allowed to have a mix of HostProcess
- containers and non-HostProcess containers). In
- addition, if HostProcess is true then HostNetwork
- must also be set to true.
+ All of a Pod's containers must have the
+ same effective HostProcess value (it is
+ not allowed to have a mix of HostProcess
+ containers and non-HostProcess containers).
+ In addition, if HostProcess is true then
+ HostNetwork must also be set to true.
type: boolean
runAsUserName:
description: The UserName in Windows to run
@@ -2911,8 +3038,7 @@ spec:
type: integer
grpc:
description: GRPC specifies an action involving
- a GRPC port. This is a beta field and requires
- enabling GRPCContainerProbe feature gate.
+ a GRPC port.
properties:
port:
description: Port number of the gRPC service.
@@ -2946,7 +3072,10 @@ spec:
header to be used in HTTP probes
properties:
name:
- description: The header field name
+ description: The header field name.
+ This will be canonicalized upon output,
+ so case-variant names will be understood
+ as the same header.
type: string
value:
description: The header field value
diff --git a/deploy/helm/crds/apps.kubeblocks.io_componentclassdefinitions.yaml b/deploy/helm/crds/apps.kubeblocks.io_componentclassdefinitions.yaml
index 9827863faa8..84569625c77 100644
--- a/deploy/helm/crds/apps.kubeblocks.io_componentclassdefinitions.yaml
+++ b/deploy/helm/crds/apps.kubeblocks.io_componentclassdefinitions.yaml
@@ -2,8 +2,7 @@ apiVersion: apiextensions.k8s.io/v1
kind: CustomResourceDefinition
metadata:
annotations:
- controller-gen.kubebuilder.io/version: v0.9.0
- creationTimestamp: null
+ controller-gen.kubebuilder.io/version: v0.12.1
labels:
app.kubernetes.io/name: kubeblocks
name: componentclassdefinitions.apps.kubeblocks.io
diff --git a/deploy/helm/crds/apps.kubeblocks.io_componentresourceconstraints.yaml b/deploy/helm/crds/apps.kubeblocks.io_componentresourceconstraints.yaml
index d96a325fe7a..0854049bfa6 100644
--- a/deploy/helm/crds/apps.kubeblocks.io_componentresourceconstraints.yaml
+++ b/deploy/helm/crds/apps.kubeblocks.io_componentresourceconstraints.yaml
@@ -2,8 +2,7 @@ apiVersion: apiextensions.k8s.io/v1
kind: CustomResourceDefinition
metadata:
annotations:
- controller-gen.kubebuilder.io/version: v0.9.0
- creationTimestamp: null
+ controller-gen.kubebuilder.io/version: v0.12.1
labels:
app.kubernetes.io/name: kubeblocks
name: componentresourceconstraints.apps.kubeblocks.io
diff --git a/deploy/helm/crds/apps.kubeblocks.io_configconstraints.yaml b/deploy/helm/crds/apps.kubeblocks.io_configconstraints.yaml
index 1d9f4c48175..e705f33b453 100644
--- a/deploy/helm/crds/apps.kubeblocks.io_configconstraints.yaml
+++ b/deploy/helm/crds/apps.kubeblocks.io_configconstraints.yaml
@@ -2,8 +2,7 @@ apiVersion: apiextensions.k8s.io/v1
kind: CustomResourceDefinition
metadata:
annotations:
- controller-gen.kubebuilder.io/version: v0.9.0
- creationTimestamp: null
+ controller-gen.kubebuilder.io/version: v0.12.1
labels:
app.kubernetes.io/name: kubeblocks
name: configconstraints.apps.kubeblocks.io
@@ -95,6 +94,7 @@ spec:
required:
- fieldPath
type: object
+ x-kubernetes-map-type: atomic
mode:
description: 'Optional: mode bits used to set permissions
on this file, must be an octal value between 0000 and
@@ -135,6 +135,7 @@ spec:
required:
- resource
type: object
+ x-kubernetes-map-type: atomic
required:
- path
type: object
@@ -371,6 +372,7 @@ spec:
are ANDed.
type: object
type: object
+ x-kubernetes-map-type: atomic
staticParameters:
description: staticParameters, list of StaticParameter, modifications
of them trigger a process restart.
diff --git a/deploy/helm/crds/apps.kubeblocks.io_configurations.yaml b/deploy/helm/crds/apps.kubeblocks.io_configurations.yaml
index 6cc4adc83b3..7dbc26db996 100644
--- a/deploy/helm/crds/apps.kubeblocks.io_configurations.yaml
+++ b/deploy/helm/crds/apps.kubeblocks.io_configurations.yaml
@@ -2,8 +2,7 @@ apiVersion: apiextensions.k8s.io/v1
kind: CustomResourceDefinition
metadata:
annotations:
- controller-gen.kubebuilder.io/version: v0.9.0
- creationTimestamp: null
+ controller-gen.kubebuilder.io/version: v0.12.1
labels:
app.kubernetes.io/name: kubeblocks
name: configurations.apps.kubeblocks.io
@@ -36,21 +35,12 @@ spec:
spec:
description: ConfigurationSpec defines the desired state of Configuration
properties:
- clusterDefRef:
- description: clusterDefRef referencing ClusterDefinition name. This
- is an immutable attribute.
- pattern: ^[a-z0-9]([a-z0-9\.\-]*[a-z0-9])?$
- type: string
clusterRef:
description: clusterRef references Cluster name.
type: string
x-kubernetes-validations:
- message: forbidden to update spec.clusterRef
rule: self == oldSelf
- clusterVerRef:
- description: clusterVerRef referencing ClusterVersion name.
- pattern: ^[a-z0-9]([a-z0-9\.\-]*[a-z0-9])?$
- type: string
componentName:
description: componentName is cluster component name.
type: string
@@ -79,6 +69,104 @@ spec:
description: configFileParams is used to set the parameters
to be updated.
type: object
+ configSpec:
+ description: configSpec is used to set the configuration template.
+ properties:
+ asEnvFrom:
+ description: 'asEnvFrom is optional: the list of containers
+ will be injected into EnvFrom.'
+ items:
+ type: string
+ type: array
+ x-kubernetes-list-type: set
+ constraintRef:
+ description: Specify the name of the referenced the configuration
+ constraints object.
+ maxLength: 63
+ pattern: ^[a-z0-9]([a-z0-9\.\-]*[a-z0-9])?$
+ type: string
+ defaultMode:
+ description: 'defaultMode is optional: mode bits used to
+ set permissions on created files by default. Must be an
+ octal value between 0000 and 0777 or a decimal value between
+ 0 and 511. YAML accepts both octal and decimal values,
+ JSON requires decimal values for mode bits. Defaults to
+ 0644. Directories within the path are not affected by
+ this setting. This might be in conflict with other options
+ that affect the file mode, like fsGroup, and the result
+ can be other mode bits set.'
+ format: int32
+ type: integer
+ keys:
+ description: Specify a list of keys. If empty, ConfigConstraint
+ takes effect for all keys in configmap.
+ items:
+ type: string
+ type: array
+ x-kubernetes-list-type: set
+ legacyRenderedConfigSpec:
+ description: 'lazyRenderedConfigSpec is optional: specify
+ the secondary rendered config spec.'
+ properties:
+ namespace:
+ default: default
+ description: Specify the namespace of the referenced
+ the configuration template ConfigMap object. An empty
+ namespace is equivalent to the "default" namespace.
+ maxLength: 63
+ pattern: ^[a-z0-9]([a-z0-9\-]*[a-z0-9])?$
+ type: string
+ policy:
+ default: none
+ description: policy defines how to merge external imported
+ templates into component templates.
+ enum:
+ - patch
+ - replace
+ - none
+ type: string
+ templateRef:
+ description: Specify the name of the referenced the
+ configuration template ConfigMap object.
+ maxLength: 63
+ pattern: ^[a-z0-9]([a-z0-9\.\-]*[a-z0-9])?$
+ type: string
+ required:
+ - templateRef
+ type: object
+ name:
+ description: Specify the name of configuration template.
+ maxLength: 63
+ pattern: ^[a-z0-9]([a-z0-9\.\-]*[a-z0-9])?$
+ type: string
+ namespace:
+ default: default
+ description: Specify the namespace of the referenced the
+ configuration template ConfigMap object. An empty namespace
+ is equivalent to the "default" namespace.
+ maxLength: 63
+ pattern: ^[a-z0-9]([a-z0-9\-]*[a-z0-9])?$
+ type: string
+ templateRef:
+ description: Specify the name of the referenced the configuration
+ template ConfigMap object.
+ maxLength: 63
+ pattern: ^[a-z0-9]([a-z0-9\.\-]*[a-z0-9])?$
+ type: string
+ volumeName:
+ description: volumeName is the volume name of PodTemplate,
+ which the configuration file produced through the configuration
+ template will be mounted to the corresponding volume.
+ Must be a DNS_LABEL name. The volume name must be defined
+ in podSpec.containers[*].volumeMounts.
+ maxLength: 63
+ pattern: ^[a-z]([a-z0-9\-]*[a-z0-9])?$
+ type: string
+ required:
+ - name
+ - templateRef
+ - volumeName
+ type: object
importTemplateRef:
description: Specify the configuration template.
properties:
@@ -124,7 +212,6 @@ spec:
- name
x-kubernetes-list-type: map
required:
- - clusterDefRef
- clusterRef
- componentName
type: object
@@ -223,6 +310,7 @@ spec:
phase:
description: phase is status of configurationItem.
enum:
+ - Creating
- Init
- Running
- Pending
diff --git a/deploy/helm/crds/apps.kubeblocks.io_opsrequests.yaml b/deploy/helm/crds/apps.kubeblocks.io_opsrequests.yaml
index 140b9cbb572..16bf1252b31 100644
--- a/deploy/helm/crds/apps.kubeblocks.io_opsrequests.yaml
+++ b/deploy/helm/crds/apps.kubeblocks.io_opsrequests.yaml
@@ -2,8 +2,7 @@ apiVersion: apiextensions.k8s.io/v1
kind: CustomResourceDefinition
metadata:
annotations:
- controller-gen.kubebuilder.io/version: v0.9.0
- creationTimestamp: null
+ controller-gen.kubebuilder.io/version: v0.12.1
labels:
app.kubernetes.io/name: kubeblocks
name: opsrequests.apps.kubeblocks.io
@@ -64,17 +63,15 @@ spec:
backupSpec:
description: backupSpec defines how to backup the cluster.
properties:
+ backupMethod:
+ description: Backup method name that is defined in backupPolicy.
+ type: string
backupName:
description: backupName is the name of the backup.
type: string
backupPolicyName:
description: Which backupPolicy is applied to perform this backup
type: string
- backupType:
- default: datafile
- description: Backup Type. datafile or logfile or snapshot. If
- not set, datafile is the default type.
- type: string
parentBackupName:
description: if backupType is incremental, parentBackupName is
required.
@@ -363,6 +360,7 @@ spec:
required:
- key
type: object
+ x-kubernetes-map-type: atomic
type: array
x-kubernetes-validations:
- message: forbidden to update spec.scriptSpec.scriptFrom.configMapRef
@@ -387,6 +385,7 @@ spec:
required:
- key
type: object
+ x-kubernetes-map-type: atomic
type: array
x-kubernetes-validations:
- message: forbidden to update spec.scriptSpec.scriptFrom.secretRef
@@ -418,6 +417,59 @@ spec:
required:
- name
type: object
+ selector:
+ description: KubeBlocks, by default, will execute the script on
+ the primary pod, with role=leader. There are some exceptions,
+ such as Redis, which does not synchronize accounts info between
+ primary and secondary. In this case, we need to execute the
+ script on all pods, matching the selector. selector indicates
+ the components on which the script is executed.
+ properties:
+ matchExpressions:
+ description: matchExpressions is a list of label selector
+ requirements. The requirements are ANDed.
+ items:
+ description: A label selector requirement is a selector
+ that contains values, a key, and an operator that relates
+ the key and values.
+ properties:
+ key:
+ description: key is the label key that the selector
+ applies to.
+ type: string
+ operator:
+ description: operator represents a key's relationship
+ to a set of values. Valid operators are In, NotIn,
+ Exists and DoesNotExist.
+ type: string
+ values:
+ description: values is an array of string values. If
+ the operator is In or NotIn, the values array must
+ be non-empty. If the operator is Exists or DoesNotExist,
+ the values array must be empty. This array is replaced
+ during a strategic merge patch.
+ items:
+ type: string
+ type: array
+ required:
+ - key
+ - operator
+ type: object
+ type: array
+ matchLabels:
+ additionalProperties:
+ type: string
+ description: matchLabels is a map of {key,value} pairs. A
+ single {key,value} in the matchLabels map is equivalent
+ to an element of matchExpressions, whose key field is "key",
+ the operator is "In", and the values array contains only
+ "value". The requirements are ANDed.
+ type: object
+ type: object
+ x-kubernetes-map-type: atomic
+ x-kubernetes-validations:
+ - message: forbidden to update spec.scriptSpec.script.selector
+ rule: self == oldSelf
required:
- componentName
type: object
@@ -564,7 +616,8 @@ spec:
description: 'Requests describes the minimum amount of compute
resources required. If Requests is omitted for a container,
it defaults to Limits if that is explicitly specified, otherwise
- to an implementation-defined value. More info: https://kubernetes.io/docs/concepts/configuration/manage-resources-containers/'
+ to an implementation-defined value. Requests cannot exceed
+ Limits. More info: https://kubernetes.io/docs/concepts/configuration/manage-resources-containers/'
type: object
required:
- componentName
@@ -867,8 +920,8 @@ spec:
description: 'Requests describes the minimum amount of compute
resources required. If Requests is omitted for a container,
it defaults to Limits if that is explicitly specified,
- otherwise to an implementation-defined value. More info:
- https://kubernetes.io/docs/concepts/configuration/manage-resources-containers/'
+ otherwise to an implementation-defined value. Requests
+ cannot exceed Limits. More info: https://kubernetes.io/docs/concepts/configuration/manage-resources-containers/'
type: object
services:
description: services records the last services of the component.
diff --git a/deploy/helm/crds/apps.kubeblocks.io_servicedescriptors.yaml b/deploy/helm/crds/apps.kubeblocks.io_servicedescriptors.yaml
index 2f0dcd73083..653b4a30527 100644
--- a/deploy/helm/crds/apps.kubeblocks.io_servicedescriptors.yaml
+++ b/deploy/helm/crds/apps.kubeblocks.io_servicedescriptors.yaml
@@ -2,8 +2,7 @@ apiVersion: apiextensions.k8s.io/v1
kind: CustomResourceDefinition
metadata:
annotations:
- controller-gen.kubebuilder.io/version: v0.9.0
- creationTimestamp: null
+ controller-gen.kubebuilder.io/version: v0.12.1
labels:
app.kubernetes.io/name: kubeblocks
name: servicedescriptors.apps.kubeblocks.io
@@ -97,6 +96,7 @@ spec:
required:
- key
type: object
+ x-kubernetes-map-type: atomic
fieldRef:
description: 'Selects a field of the pod: supports metadata.name,
metadata.namespace, `metadata.labels['''']`, `metadata.annotations['''']`,
@@ -114,6 +114,7 @@ spec:
required:
- fieldPath
type: object
+ x-kubernetes-map-type: atomic
resourceFieldRef:
description: 'Selects a resource of the container: only
resources limits and requests (limits.cpu, limits.memory,
@@ -138,6 +139,7 @@ spec:
required:
- resource
type: object
+ x-kubernetes-map-type: atomic
secretKeyRef:
description: Selects a key of a secret in the pod's namespace
properties:
@@ -157,6 +159,7 @@ spec:
required:
- key
type: object
+ x-kubernetes-map-type: atomic
type: object
type: object
username:
@@ -196,6 +199,7 @@ spec:
required:
- key
type: object
+ x-kubernetes-map-type: atomic
fieldRef:
description: 'Selects a field of the pod: supports metadata.name,
metadata.namespace, `metadata.labels['''']`, `metadata.annotations['''']`,
@@ -213,6 +217,7 @@ spec:
required:
- fieldPath
type: object
+ x-kubernetes-map-type: atomic
resourceFieldRef:
description: 'Selects a resource of the container: only
resources limits and requests (limits.cpu, limits.memory,
@@ -237,6 +242,7 @@ spec:
required:
- resource
type: object
+ x-kubernetes-map-type: atomic
secretKeyRef:
description: Selects a key of a secret in the pod's namespace
properties:
@@ -256,6 +262,7 @@ spec:
required:
- key
type: object
+ x-kubernetes-map-type: atomic
type: object
type: object
type: object
@@ -294,6 +301,7 @@ spec:
required:
- key
type: object
+ x-kubernetes-map-type: atomic
fieldRef:
description: 'Selects a field of the pod: supports metadata.name,
metadata.namespace, `metadata.labels['''']`, `metadata.annotations['''']`,
@@ -311,6 +319,7 @@ spec:
required:
- fieldPath
type: object
+ x-kubernetes-map-type: atomic
resourceFieldRef:
description: 'Selects a resource of the container: only resources
limits and requests (limits.cpu, limits.memory, limits.ephemeral-storage,
@@ -335,6 +344,7 @@ spec:
required:
- resource
type: object
+ x-kubernetes-map-type: atomic
secretKeyRef:
description: Selects a key of a secret in the pod's namespace
properties:
@@ -353,6 +363,7 @@ spec:
required:
- key
type: object
+ x-kubernetes-map-type: atomic
type: object
type: object
port:
@@ -390,6 +401,7 @@ spec:
required:
- key
type: object
+ x-kubernetes-map-type: atomic
fieldRef:
description: 'Selects a field of the pod: supports metadata.name,
metadata.namespace, `metadata.labels['''']`, `metadata.annotations['''']`,
@@ -407,6 +419,7 @@ spec:
required:
- fieldPath
type: object
+ x-kubernetes-map-type: atomic
resourceFieldRef:
description: 'Selects a resource of the container: only resources
limits and requests (limits.cpu, limits.memory, limits.ephemeral-storage,
@@ -431,6 +444,7 @@ spec:
required:
- resource
type: object
+ x-kubernetes-map-type: atomic
secretKeyRef:
description: Selects a key of a secret in the pod's namespace
properties:
@@ -449,6 +463,7 @@ spec:
required:
- key
type: object
+ x-kubernetes-map-type: atomic
type: object
type: object
serviceKind:
diff --git a/deploy/helm/crds/dataprotection.kubeblocks.io_actionsets.yaml b/deploy/helm/crds/dataprotection.kubeblocks.io_actionsets.yaml
new file mode 100644
index 00000000000..85c9df43f48
--- /dev/null
+++ b/deploy/helm/crds/dataprotection.kubeblocks.io_actionsets.yaml
@@ -0,0 +1,554 @@
+apiVersion: apiextensions.k8s.io/v1
+kind: CustomResourceDefinition
+metadata:
+ annotations:
+ controller-gen.kubebuilder.io/version: v0.12.1
+ labels:
+ app.kubernetes.io/name: kubeblocks
+ name: actionsets.dataprotection.kubeblocks.io
+spec:
+ group: dataprotection.kubeblocks.io
+ names:
+ categories:
+ - kubeblocks
+ kind: ActionSet
+ listKind: ActionSetList
+ plural: actionsets
+ shortNames:
+ - as
+ singular: actionset
+ scope: Cluster
+ versions:
+ - additionalPrinterColumns:
+ - jsonPath: .spec.backupType
+ name: BACKUP-TYPE
+ type: string
+ - jsonPath: .status.phase
+ name: STATUS
+ type: string
+ - jsonPath: .metadata.creationTimestamp
+ name: AGE
+ type: date
+ name: v1alpha1
+ schema:
+ openAPIV3Schema:
+ description: ActionSet is the Schema for the actionsets API
+ properties:
+ apiVersion:
+ description: 'APIVersion defines the versioned schema of this representation
+ of an object. Servers should convert recognized schemas to the latest
+ internal value, and may reject unrecognized values. More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#resources'
+ type: string
+ kind:
+ description: 'Kind is a string value representing the REST resource this
+ object represents. Servers may infer this from the endpoint the client
+ submits requests to. Cannot be updated. In CamelCase. More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#types-kinds'
+ type: string
+ metadata:
+ type: object
+ spec:
+ description: ActionSetSpec defines the desired state of ActionSet
+ properties:
+ backup:
+ description: backup specifies the backup action.
+ properties:
+ backupData:
+ description: backupData specifies the backup data action.
+ properties:
+ command:
+ description: command specifies the commands to back up the
+ volume data.
+ items:
+ type: string
+ type: array
+ image:
+ description: image specifies the image of backup container.
+ type: string
+ onError:
+ default: Fail
+ description: OnError specifies how should behave if it encounters
+ an error executing this action.
+ enum:
+ - Continue
+ - Fail
+ type: string
+ runOnTargetPodNode:
+ default: false
+ description: runOnTargetPodNode specifies whether to run the
+ job workload on the target pod node. If backup container
+ should mount the target pod's volume, this field should
+ be set to true.
+ type: boolean
+ syncProgress:
+ description: syncProgress specifies whether to sync the backup
+ progress and its interval seconds.
+ properties:
+ enabled:
+ description: enabled specifies whether to sync the backup
+ progress. If enabled, a sidecar container will be created
+ to sync the backup progress to the Backup CR status.
+ type: boolean
+ intervalSeconds:
+ default: 60
+ description: intervalSeconds specifies the interval seconds
+ to sync the backup progress.
+ format: int32
+ type: integer
+ type: object
+ required:
+ - command
+ - image
+ type: object
+ postBackup:
+ description: postBackup specifies a hook that should be executed
+ after the backup.
+ items:
+ description: ActionSpec defines an action that should be executed.
+ Only one of the fields may be set.
+ properties:
+ exec:
+ description: exec specifies the action should be executed
+ by the pod exec API in a container.
+ properties:
+ command:
+ description: Command is the command and arguments to
+ execute.
+ items:
+ type: string
+ minItems: 1
+ type: array
+ container:
+ description: container is the container in the pod where
+ the command should be executed. If not specified,
+ the pod's first container is used.
+ type: string
+ onError:
+ default: Fail
+ description: OnError specifies how should behave if
+ it encounters an error executing this action.
+ enum:
+ - Continue
+ - Fail
+ type: string
+ timeout:
+ description: Timeout defines the maximum amount of time
+ should wait for the hook to complete before considering
+ the execution a failure.
+ type: string
+ required:
+ - command
+ type: object
+ job:
+ description: job specifies the action should be executed
+ by a Kubernetes Job.
+ properties:
+ command:
+ description: command specifies the commands to back
+ up the volume data.
+ items:
+ type: string
+ type: array
+ image:
+ description: image specifies the image of backup container.
+ type: string
+ onError:
+ default: Fail
+ description: OnError specifies how should behave if
+ it encounters an error executing this action.
+ enum:
+ - Continue
+ - Fail
+ type: string
+ runOnTargetPodNode:
+ default: false
+ description: runOnTargetPodNode specifies whether to
+ run the job workload on the target pod node. If backup
+ container should mount the target pod's volume, this
+ field should be set to true.
+ type: boolean
+ required:
+ - command
+ - image
+ type: object
+ type: object
+ type: array
+ preBackup:
+ description: preBackup specifies a hook that should be executed
+ before the backup.
+ items:
+ description: ActionSpec defines an action that should be executed.
+ Only one of the fields may be set.
+ properties:
+ exec:
+ description: exec specifies the action should be executed
+ by the pod exec API in a container.
+ properties:
+ command:
+ description: Command is the command and arguments to
+ execute.
+ items:
+ type: string
+ minItems: 1
+ type: array
+ container:
+ description: container is the container in the pod where
+ the command should be executed. If not specified,
+ the pod's first container is used.
+ type: string
+ onError:
+ default: Fail
+ description: OnError specifies how should behave if
+ it encounters an error executing this action.
+ enum:
+ - Continue
+ - Fail
+ type: string
+ timeout:
+ description: Timeout defines the maximum amount of time
+ should wait for the hook to complete before considering
+ the execution a failure.
+ type: string
+ required:
+ - command
+ type: object
+ job:
+ description: job specifies the action should be executed
+ by a Kubernetes Job.
+ properties:
+ command:
+ description: command specifies the commands to back
+ up the volume data.
+ items:
+ type: string
+ type: array
+ image:
+ description: image specifies the image of backup container.
+ type: string
+ onError:
+ default: Fail
+ description: OnError specifies how should behave if
+ it encounters an error executing this action.
+ enum:
+ - Continue
+ - Fail
+ type: string
+ runOnTargetPodNode:
+ default: false
+ description: runOnTargetPodNode specifies whether to
+ run the job workload on the target pod node. If backup
+ container should mount the target pod's volume, this
+ field should be set to true.
+ type: boolean
+ required:
+ - command
+ - image
+ type: object
+ type: object
+ type: array
+ type: object
+ backupType:
+ allOf:
+ - enum:
+ - Full
+ - Incremental
+ - Differential
+ - Continuous
+ - enum:
+ - Full
+ - Incremental
+ - Differential
+ - Continuous
+ default: Full
+ description: 'backupType specifies the backup type, supported values:
+ Full, Continuous. Full means full backup. Incremental means back
+ up data that have changed since the last backup (full or incremental).
+ Differential means back up data that have changed since the last
+ full backup. Continuous will back up the transaction log continuously,
+ the PITR (Point in Time Recovery). can be performed based on the
+ continuous backup and full backup.'
+ type: string
+ env:
+ description: List of environment variables to set in the container.
+ items:
+ description: EnvVar represents an environment variable present in
+ a Container.
+ properties:
+ name:
+ description: Name of the environment variable. Must be a C_IDENTIFIER.
+ type: string
+ value:
+ description: 'Variable references $(VAR_NAME) are expanded using
+ the previously defined environment variables in the container
+ and any service environment variables. If a variable cannot
+ be resolved, the reference in the input string will be unchanged.
+ Double $$ are reduced to a single $, which allows for escaping
+ the $(VAR_NAME) syntax: i.e. "$$(VAR_NAME)" will produce the
+ string literal "$(VAR_NAME)". Escaped references will never
+ be expanded, regardless of whether the variable exists or
+ not. Defaults to "".'
+ type: string
+ valueFrom:
+ description: Source for the environment variable's value. Cannot
+ be used if value is not empty.
+ properties:
+ configMapKeyRef:
+ description: Selects a key of a ConfigMap.
+ properties:
+ key:
+ description: The key to select.
+ type: string
+ name:
+ description: 'Name of the referent. More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names
+ TODO: Add other useful fields. apiVersion, kind, uid?'
+ type: string
+ optional:
+ description: Specify whether the ConfigMap or its key
+ must be defined
+ type: boolean
+ required:
+ - key
+ type: object
+ x-kubernetes-map-type: atomic
+ fieldRef:
+ description: 'Selects a field of the pod: supports metadata.name,
+ metadata.namespace, `metadata.labels['''']`, `metadata.annotations['''']`,
+ spec.nodeName, spec.serviceAccountName, status.hostIP,
+ status.podIP, status.podIPs.'
+ properties:
+ apiVersion:
+ description: Version of the schema the FieldPath is
+ written in terms of, defaults to "v1".
+ type: string
+ fieldPath:
+ description: Path of the field to select in the specified
+ API version.
+ type: string
+ required:
+ - fieldPath
+ type: object
+ x-kubernetes-map-type: atomic
+ resourceFieldRef:
+ description: 'Selects a resource of the container: only
+ resources limits and requests (limits.cpu, limits.memory,
+ limits.ephemeral-storage, requests.cpu, requests.memory
+ and requests.ephemeral-storage) are currently supported.'
+ properties:
+ containerName:
+ description: 'Container name: required for volumes,
+ optional for env vars'
+ type: string
+ divisor:
+ anyOf:
+ - type: integer
+ - type: string
+ description: Specifies the output format of the exposed
+ resources, defaults to "1"
+ pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$
+ x-kubernetes-int-or-string: true
+ resource:
+ description: 'Required: resource to select'
+ type: string
+ required:
+ - resource
+ type: object
+ x-kubernetes-map-type: atomic
+ secretKeyRef:
+ description: Selects a key of a secret in the pod's namespace
+ properties:
+ key:
+ description: The key of the secret to select from. Must
+ be a valid secret key.
+ type: string
+ name:
+ description: 'Name of the referent. More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names
+ TODO: Add other useful fields. apiVersion, kind, uid?'
+ type: string
+ optional:
+ description: Specify whether the Secret or its key must
+ be defined
+ type: boolean
+ required:
+ - key
+ type: object
+ x-kubernetes-map-type: atomic
+ type: object
+ required:
+ - name
+ type: object
+ type: array
+ x-kubernetes-preserve-unknown-fields: true
+ envFrom:
+ description: List of sources to populate environment variables in
+ the container. The keys defined within a source must be a C_IDENTIFIER.
+ All invalid keys will be reported as an event when the container
+ is starting. When a key exists in multiple sources, the value associated
+ with the last source will take precedence. Values defined by an
+ Env with a duplicate key will take precedence. Cannot be updated.
+ items:
+ description: EnvFromSource represents the source of a set of ConfigMaps
+ properties:
+ configMapRef:
+ description: The ConfigMap to select from
+ properties:
+ name:
+ description: 'Name of the referent. More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names
+ TODO: Add other useful fields. apiVersion, kind, uid?'
+ type: string
+ optional:
+ description: Specify whether the ConfigMap must be defined
+ type: boolean
+ type: object
+ x-kubernetes-map-type: atomic
+ prefix:
+ description: An optional identifier to prepend to each key in
+ the ConfigMap. Must be a C_IDENTIFIER.
+ type: string
+ secretRef:
+ description: The Secret to select from
+ properties:
+ name:
+ description: 'Name of the referent. More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names
+ TODO: Add other useful fields. apiVersion, kind, uid?'
+ type: string
+ optional:
+ description: Specify whether the Secret must be defined
+ type: boolean
+ type: object
+ x-kubernetes-map-type: atomic
+ type: object
+ type: array
+ x-kubernetes-preserve-unknown-fields: true
+ restore:
+ description: restore specifies the restore action.
+ properties:
+ postReady:
+ description: postReady specifies the action to execute after the
+ data is ready.
+ items:
+ description: ActionSpec defines an action that should be executed.
+ Only one of the fields may be set.
+ properties:
+ exec:
+ description: exec specifies the action should be executed
+ by the pod exec API in a container.
+ properties:
+ command:
+ description: Command is the command and arguments to
+ execute.
+ items:
+ type: string
+ minItems: 1
+ type: array
+ container:
+ description: container is the container in the pod where
+ the command should be executed. If not specified,
+ the pod's first container is used.
+ type: string
+ onError:
+ default: Fail
+ description: OnError specifies how should behave if
+ it encounters an error executing this action.
+ enum:
+ - Continue
+ - Fail
+ type: string
+ timeout:
+ description: Timeout defines the maximum amount of time
+ should wait for the hook to complete before considering
+ the execution a failure.
+ type: string
+ required:
+ - command
+ type: object
+ job:
+ description: job specifies the action should be executed
+ by a Kubernetes Job.
+ properties:
+ command:
+ description: command specifies the commands to back
+ up the volume data.
+ items:
+ type: string
+ type: array
+ image:
+ description: image specifies the image of backup container.
+ type: string
+ onError:
+ default: Fail
+ description: OnError specifies how should behave if
+ it encounters an error executing this action.
+ enum:
+ - Continue
+ - Fail
+ type: string
+ runOnTargetPodNode:
+ default: false
+ description: runOnTargetPodNode specifies whether to
+ run the job workload on the target pod node. If backup
+ container should mount the target pod's volume, this
+ field should be set to true.
+ type: boolean
+ required:
+ - command
+ - image
+ type: object
+ type: object
+ type: array
+ prepareData:
+ description: prepareData specifies the action to prepare data.
+ properties:
+ command:
+ description: command specifies the commands to back up the
+ volume data.
+ items:
+ type: string
+ type: array
+ image:
+ description: image specifies the image of backup container.
+ type: string
+ onError:
+ default: Fail
+ description: OnError specifies how should behave if it encounters
+ an error executing this action.
+ enum:
+ - Continue
+ - Fail
+ type: string
+ runOnTargetPodNode:
+ default: false
+ description: runOnTargetPodNode specifies whether to run the
+ job workload on the target pod node. If backup container
+ should mount the target pod's volume, this field should
+ be set to true.
+ type: boolean
+ required:
+ - command
+ - image
+ type: object
+ type: object
+ required:
+ - backupType
+ type: object
+ status:
+ description: ActionSetStatus defines the observed state of ActionSet
+ properties:
+ message:
+ description: A human-readable message indicating details about why
+ the ActionSet is in this phase.
+ type: string
+ observedGeneration:
+ description: generation number
+ format: int64
+ type: integer
+ phase:
+ description: phase - in list of [Available,Unavailable]
+ enum:
+ - Available
+ - Unavailable
+ type: string
+ type: object
+ type: object
+ served: true
+ storage: true
+ subresources:
+ status: {}
diff --git a/deploy/helm/crds/dataprotection.kubeblocks.io_backuppolicies.yaml b/deploy/helm/crds/dataprotection.kubeblocks.io_backuppolicies.yaml
index f3261eca785..ec61183f863 100644
--- a/deploy/helm/crds/dataprotection.kubeblocks.io_backuppolicies.yaml
+++ b/deploy/helm/crds/dataprotection.kubeblocks.io_backuppolicies.yaml
@@ -2,8 +2,7 @@ apiVersion: apiextensions.k8s.io/v1
kind: CustomResourceDefinition
metadata:
annotations:
- controller-gen.kubebuilder.io/version: v0.9.0
- creationTimestamp: null
+ controller-gen.kubebuilder.io/version: v0.12.1
labels:
app.kubernetes.io/name: kubeblocks
name: backuppolicies.dataprotection.kubeblocks.io
@@ -21,20 +20,19 @@ spec:
scope: Namespaced
versions:
- additionalPrinterColumns:
+ - jsonPath: .spec.backupRepoName
+ name: BACKUP-REPO
+ type: string
- jsonPath: .status.phase
name: STATUS
type: string
- - jsonPath: .status.lastScheduleTime
- name: LAST SCHEDULE
- type: string
- jsonPath: .metadata.creationTimestamp
name: AGE
type: date
name: v1alpha1
schema:
openAPIV3Schema:
- description: BackupPolicy is the Schema for the backuppolicies API (defined
- by User)
+ description: BackupPolicy is the Schema for the backuppolicies API.
properties:
apiVersion:
description: 'APIVersion defines the versioned schema of this representation
@@ -51,540 +49,389 @@ spec:
spec:
description: BackupPolicySpec defines the desired state of BackupPolicy
properties:
- datafile:
- description: the policy for datafile backup.
- properties:
- backupRepoName:
- description: refer to BackupRepo and the backup data will be stored
- in the corresponding repo.
- pattern: ^[a-z0-9]([a-z0-9\.\-]*[a-z0-9])?$
- type: string
- backupStatusUpdates:
- description: define how to update metadata for backup status.
- items:
- properties:
- containerName:
- description: which container name that kubectl can execute.
- type: string
- path:
- description: 'specify the json path of backup object for
- patch. example: manifests.backupLog -- means patch the
- backup json path of status.manifests.backupLog.'
- type: string
- script:
- description: the shell Script commands to collect backup
- status metadata. The script must exist in the container
- of ContainerName and the output format must be set to
- JSON. Note that outputting to stderr may cause the result
- format to not be in JSON.
- type: string
- updateStage:
- description: 'when to update the backup status, pre: before
- backup, post: after backup'
- enum:
- - pre
- - post
- type: string
- useTargetPodServiceAccount:
- description: useTargetPodServiceAccount defines whether
- this job requires the service account of the backup target
- pod. if true, will use the service account of the backup
- target pod. otherwise, will use the system service account.
- type: boolean
- required:
- - updateStage
- type: object
- type: array
- backupToolName:
- description: which backup tool to perform database backup, only
- support one tool.
- pattern: ^[a-z0-9]([a-z0-9\.\-]*[a-z0-9])?$
- type: string
- backupsHistoryLimit:
- default: 7
- description: the number of automatic backups to retain. Value
- must be non-negative integer. 0 means NO limit on the number
- of backups.
- format: int32
- type: integer
- onFailAttempted:
- description: count of backup stop retries on fail.
- format: int32
- type: integer
- persistentVolumeClaim:
- description: refer to PersistentVolumeClaim and the backup data
- will be stored in the corresponding persistent volume.
- properties:
- createPolicy:
- default: IfNotPresent
- description: 'createPolicy defines the policy for creating
- the PersistentVolumeClaim, enum values: - Never: do nothing
- if the PersistentVolumeClaim not exists. - IfNotPresent:
- create the PersistentVolumeClaim if not present and the
- accessModes only contains ''ReadWriteMany''.'
- enum:
- - IfNotPresent
- - Never
- type: string
- initCapacity:
- anyOf:
- - type: integer
- - type: string
- description: initCapacity represents the init storage size
- of the PersistentVolumeClaim which should be created if
- not exist. and the default value is 100Gi if it is empty.
- pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$
- x-kubernetes-int-or-string: true
- name:
- description: the name of PersistentVolumeClaim to store backup
- data.
- type: string
- persistentVolumeConfigMap:
- description: 'persistentVolumeConfigMap references the configmap
- which contains a persistentVolume template. key must be
- "persistentVolume" and value is the "PersistentVolume" struct.
- support the following built-in Objects: - $(GENERATE_NAME):
- generate a specific format "`PVC NAME`-`PVC NAMESPACE`".
- if the PersistentVolumeClaim not exists and CreatePolicy
- is "IfNotPresent", the controller will create it by this
- template. this is a mutually exclusive setting with "storageClassName".'
+ backoffLimit:
+ description: Specifies the number of retries before marking the backup
+ failed.
+ format: int32
+ maximum: 10
+ minimum: 0
+ type: integer
+ backupMethods:
+ description: backupMethods defines the backup methods.
+ items:
+ description: BackupMethod defines the backup method.
+ properties:
+ actionSetName:
+ description: actionSetName refers to the ActionSet object that
+ defines the backup actions. For volume snapshot backup, the
+ actionSet is not required, the controller will use the CSI
+ volume snapshotter to create the snapshot.
+ type: string
+ env:
+ description: env specifies the environment variables for the
+ backup workload.
+ items:
+ description: EnvVar represents an environment variable present
+ in a Container.
properties:
name:
- description: the name of the persistentVolume ConfigMap.
- pattern: ^[a-z0-9]([a-z0-9\.\-]*[a-z0-9])?$
+ description: Name of the environment variable. Must be
+ a C_IDENTIFIER.
type: string
- namespace:
- description: the namespace of the persistentVolume ConfigMap.
- pattern: ^[a-z0-9]([a-z0-9\-]*[a-z0-9])?$
+ value:
+ description: 'Variable references $(VAR_NAME) are expanded
+ using the previously defined environment variables in
+ the container and any service environment variables.
+ If a variable cannot be resolved, the reference in the
+ input string will be unchanged. Double $$ are reduced
+ to a single $, which allows for escaping the $(VAR_NAME)
+ syntax: i.e. "$$(VAR_NAME)" will produce the string
+ literal "$(VAR_NAME)". Escaped references will never
+ be expanded, regardless of whether the variable exists
+ or not. Defaults to "".'
type: string
- required:
- - name
- - namespace
- type: object
- storageClassName:
- description: storageClassName is the name of the StorageClass
- required by the claim.
- pattern: ^[a-z0-9]([a-z0-9\.\-]*[a-z0-9])?$
- type: string
- type: object
- target:
- description: target database cluster for backup.
- properties:
- labelsSelector:
- description: labelsSelector is used to find matching pods.
- Pods that match this label selector are counted to determine
- the number of pods in their corresponding topology domain.
- properties:
- matchExpressions:
- description: matchExpressions is a list of label selector
- requirements. The requirements are ANDed.
- items:
- description: A label selector requirement is a selector
- that contains values, a key, and an operator that
- relates the key and values.
- properties:
- key:
- description: key is the label key that the selector
- applies to.
- type: string
- operator:
- description: operator represents a key's relationship
- to a set of values. Valid operators are In, NotIn,
- Exists and DoesNotExist.
- type: string
- values:
- description: values is an array of string values.
- If the operator is In or NotIn, the values array
- must be non-empty. If the operator is Exists or
- DoesNotExist, the values array must be empty.
- This array is replaced during a strategic merge
- patch.
- items:
+ valueFrom:
+ description: Source for the environment variable's value.
+ Cannot be used if value is not empty.
+ properties:
+ configMapKeyRef:
+ description: Selects a key of a ConfigMap.
+ properties:
+ key:
+ description: The key to select.
type: string
- type: array
- required:
- - key
- - operator
- type: object
- type: array
- matchLabels:
- additionalProperties:
- type: string
- description: matchLabels is a map of {key,value} pairs.
- A single {key,value} in the matchLabels map is equivalent
- to an element of matchExpressions, whose key field is
- "key", the operator is "In", and the values array contains
- only "value". The requirements are ANDed.
+ name:
+ description: 'Name of the referent. More info:
+ https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names
+ TODO: Add other useful fields. apiVersion, kind,
+ uid?'
+ type: string
+ optional:
+ description: Specify whether the ConfigMap or
+ its key must be defined
+ type: boolean
+ required:
+ - key
+ type: object
+ x-kubernetes-map-type: atomic
+ fieldRef:
+ description: 'Selects a field of the pod: supports
+ metadata.name, metadata.namespace, `metadata.labels['''']`,
+ `metadata.annotations['''']`, spec.nodeName,
+ spec.serviceAccountName, status.hostIP, status.podIP,
+ status.podIPs.'
+ properties:
+ apiVersion:
+ description: Version of the schema the FieldPath
+ is written in terms of, defaults to "v1".
+ type: string
+ fieldPath:
+ description: Path of the field to select in the
+ specified API version.
+ type: string
+ required:
+ - fieldPath
+ type: object
+ x-kubernetes-map-type: atomic
+ resourceFieldRef:
+ description: 'Selects a resource of the container:
+ only resources limits and requests (limits.cpu,
+ limits.memory, limits.ephemeral-storage, requests.cpu,
+ requests.memory and requests.ephemeral-storage)
+ are currently supported.'
+ properties:
+ containerName:
+ description: 'Container name: required for volumes,
+ optional for env vars'
+ type: string
+ divisor:
+ anyOf:
+ - type: integer
+ - type: string
+ description: Specifies the output format of the
+ exposed resources, defaults to "1"
+ pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$
+ x-kubernetes-int-or-string: true
+ resource:
+ description: 'Required: resource to select'
+ type: string
+ required:
+ - resource
+ type: object
+ x-kubernetes-map-type: atomic
+ secretKeyRef:
+ description: Selects a key of a secret in the pod's
+ namespace
+ properties:
+ key:
+ description: The key of the secret to select from. Must
+ be a valid secret key.
+ type: string
+ name:
+ description: 'Name of the referent. More info:
+ https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names
+ TODO: Add other useful fields. apiVersion, kind,
+ uid?'
+ type: string
+ optional:
+ description: Specify whether the Secret or its
+ key must be defined
+ type: boolean
+ required:
+ - key
+ type: object
+ x-kubernetes-map-type: atomic
type: object
- type: object
- x-kubernetes-preserve-unknown-fields: true
- secret:
- description: secret is used to connect to the target database
- cluster. If not set, secret will be inherited from backup
- policy template. if still not set, the controller will check
- if any system account for dataprotection has been created.
- properties:
- name:
- description: the secret name
- pattern: ^[a-z0-9]([a-z0-9\.\-]*[a-z0-9])?$
- type: string
- passwordKey:
- default: password
- description: passwordKey the map key of the password in
- the connection credential secret
- type: string
- usernameKey:
- default: username
- description: usernameKey the map key of the user in the
- connection credential secret
- type: string
required:
- name
type: object
- required:
- - labelsSelector
- type: object
- required:
- - target
- type: object
- logfile:
- description: the policy for logfile backup.
- properties:
- backupRepoName:
- description: refer to BackupRepo and the backup data will be stored
- in the corresponding repo.
- pattern: ^[a-z0-9]([a-z0-9\.\-]*[a-z0-9])?$
- type: string
- backupStatusUpdates:
- description: define how to update metadata for backup status.
- items:
+ type: array
+ name:
+ description: the name of backup method.
+ pattern: ^[a-z0-9]([a-z0-9\.\-]*[a-z0-9])?$
+ type: string
+ runtimeSettings:
+ description: runtimeSettings specifies runtime settings for
+ the backup workload container.
properties:
- containerName:
- description: which container name that kubectl can execute.
- type: string
- path:
- description: 'specify the json path of backup object for
- patch. example: manifests.backupLog -- means patch the
- backup json path of status.manifests.backupLog.'
- type: string
- script:
- description: the shell Script commands to collect backup
- status metadata. The script must exist in the container
- of ContainerName and the output format must be set to
- JSON. Note that outputting to stderr may cause the result
- format to not be in JSON.
- type: string
- updateStage:
- description: 'when to update the backup status, pre: before
- backup, post: after backup'
- enum:
- - pre
- - post
- type: string
- useTargetPodServiceAccount:
- description: useTargetPodServiceAccount defines whether
- this job requires the service account of the backup target
- pod. if true, will use the service account of the backup
- target pod. otherwise, will use the system service account.
- type: boolean
- required:
- - updateStage
- type: object
- type: array
- backupToolName:
- description: which backup tool to perform database backup, only
- support one tool.
- pattern: ^[a-z0-9]([a-z0-9\.\-]*[a-z0-9])?$
- type: string
- backupsHistoryLimit:
- default: 7
- description: the number of automatic backups to retain. Value
- must be non-negative integer. 0 means NO limit on the number
- of backups.
- format: int32
- type: integer
- onFailAttempted:
- description: count of backup stop retries on fail.
- format: int32
- type: integer
- persistentVolumeClaim:
- description: refer to PersistentVolumeClaim and the backup data
- will be stored in the corresponding persistent volume.
- properties:
- createPolicy:
- default: IfNotPresent
- description: 'createPolicy defines the policy for creating
- the PersistentVolumeClaim, enum values: - Never: do nothing
- if the PersistentVolumeClaim not exists. - IfNotPresent:
- create the PersistentVolumeClaim if not present and the
- accessModes only contains ''ReadWriteMany''.'
- enum:
- - IfNotPresent
- - Never
- type: string
- initCapacity:
- anyOf:
- - type: integer
- - type: string
- description: initCapacity represents the init storage size
- of the PersistentVolumeClaim which should be created if
- not exist. and the default value is 100Gi if it is empty.
- pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$
- x-kubernetes-int-or-string: true
- name:
- description: the name of PersistentVolumeClaim to store backup
- data.
- type: string
- persistentVolumeConfigMap:
- description: 'persistentVolumeConfigMap references the configmap
- which contains a persistentVolume template. key must be
- "persistentVolume" and value is the "PersistentVolume" struct.
- support the following built-in Objects: - $(GENERATE_NAME):
- generate a specific format "`PVC NAME`-`PVC NAMESPACE`".
- if the PersistentVolumeClaim not exists and CreatePolicy
- is "IfNotPresent", the controller will create it by this
- template. this is a mutually exclusive setting with "storageClassName".'
- properties:
- name:
- description: the name of the persistentVolume ConfigMap.
- pattern: ^[a-z0-9]([a-z0-9\.\-]*[a-z0-9])?$
- type: string
- namespace:
- description: the namespace of the persistentVolume ConfigMap.
- pattern: ^[a-z0-9]([a-z0-9\-]*[a-z0-9])?$
- type: string
- required:
- - name
- - namespace
- type: object
- storageClassName:
- description: storageClassName is the name of the StorageClass
- required by the claim.
- pattern: ^[a-z0-9]([a-z0-9\.\-]*[a-z0-9])?$
- type: string
- type: object
- target:
- description: target database cluster for backup.
- properties:
- labelsSelector:
- description: labelsSelector is used to find matching pods.
- Pods that match this label selector are counted to determine
- the number of pods in their corresponding topology domain.
- properties:
- matchExpressions:
- description: matchExpressions is a list of label selector
- requirements. The requirements are ANDed.
- items:
- description: A label selector requirement is a selector
- that contains values, a key, and an operator that
- relates the key and values.
- properties:
- key:
- description: key is the label key that the selector
- applies to.
- type: string
- operator:
- description: operator represents a key's relationship
- to a set of values. Valid operators are In, NotIn,
- Exists and DoesNotExist.
- type: string
- values:
- description: values is an array of string values.
- If the operator is In or NotIn, the values array
- must be non-empty. If the operator is Exists or
- DoesNotExist, the values array must be empty.
- This array is replaced during a strategic merge
- patch.
- items:
+ resources:
+ description: 'resources specifies the resource required
+ by container. More info: https://kubernetes.io/docs/concepts/configuration/manage-compute-resources-container/'
+ properties:
+ claims:
+ description: "Claims lists the names of resources, defined
+ in spec.resourceClaims, that are used by this container.
+ \n This is an alpha field and requires enabling the
+ DynamicResourceAllocation feature gate. \n This field
+ is immutable. It can only be set for containers."
+ items:
+ description: ResourceClaim references one entry in
+ PodSpec.ResourceClaims.
+ properties:
+ name:
+ description: Name must match the name of one entry
+ in pod.spec.resourceClaims of the Pod where
+ this field is used. It makes that resource available
+ inside a container.
type: string
- type: array
- required:
- - key
- - operator
+ required:
+ - name
+ type: object
+ type: array
+ x-kubernetes-list-map-keys:
+ - name
+ x-kubernetes-list-type: map
+ limits:
+ additionalProperties:
+ anyOf:
+ - type: integer
+ - type: string
+ pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$
+ x-kubernetes-int-or-string: true
+ description: 'Limits describes the maximum amount of
+ compute resources allowed. More info: https://kubernetes.io/docs/concepts/configuration/manage-resources-containers/'
type: object
- type: array
- matchLabels:
- additionalProperties:
- type: string
- description: matchLabels is a map of {key,value} pairs.
- A single {key,value} in the matchLabels map is equivalent
- to an element of matchExpressions, whose key field is
- "key", the operator is "In", and the values array contains
- only "value". The requirements are ANDed.
+ requests:
+ additionalProperties:
+ anyOf:
+ - type: integer
+ - type: string
+ pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$
+ x-kubernetes-int-or-string: true
+ description: 'Requests describes the minimum amount
+ of compute resources required. If Requests is omitted
+ for a container, it defaults to Limits if that is
+ explicitly specified, otherwise to an implementation-defined
+ value. Requests cannot exceed Limits. More info: https://kubernetes.io/docs/concepts/configuration/manage-resources-containers/'
+ type: object
+ type: object
+ type: object
+ snapshotVolumes:
+ default: false
+ description: snapshotVolumes specifies whether to take snapshots
+ of persistent volumes. if true, the BackupScript is not required,
+ the controller will use the CSI volume snapshotter to create
+ the snapshot.
+ type: boolean
+ targetVolumes:
+ description: targetVolumes specifies which volumes from the
+ target should be mounted in the backup workload.
+ properties:
+ volumeMounts:
+ description: volumeMounts specifies the mount for the volumes
+ specified in `Volumes` section.
+ items:
+ description: VolumeMount describes a mounting of a Volume
+ within a container.
+ properties:
+ mountPath:
+ description: Path within the container at which the
+ volume should be mounted. Must not contain ':'.
+ type: string
+ mountPropagation:
+ description: mountPropagation determines how mounts
+ are propagated from the host to container and the
+ other way around. When not set, MountPropagationNone
+ is used. This field is beta in 1.10.
+ type: string
+ name:
+ description: This must match the Name of a Volume.
+ type: string
+ readOnly:
+ description: Mounted read-only if true, read-write
+ otherwise (false or unspecified). Defaults to false.
+ type: boolean
+ subPath:
+ description: Path within the volume from which the
+ container's volume should be mounted. Defaults to
+ "" (volume's root).
+ type: string
+ subPathExpr:
+ description: Expanded path within the volume from
+ which the container's volume should be mounted.
+ Behaves similarly to SubPath but environment variable
+ references $(VAR_NAME) are expanded using the container's
+ environment. Defaults to "" (volume's root). SubPathExpr
+ and SubPath are mutually exclusive.
+ type: string
+ required:
+ - mountPath
+ - name
type: object
- type: object
- x-kubernetes-preserve-unknown-fields: true
- secret:
- description: secret is used to connect to the target database
- cluster. If not set, secret will be inherited from backup
- policy template. if still not set, the controller will check
- if any system account for dataprotection has been created.
- properties:
- name:
- description: the secret name
- pattern: ^[a-z0-9]([a-z0-9\.\-]*[a-z0-9])?$
+ type: array
+ volumes:
+ description: Volumes indicates the list of volumes of targeted
+ application that should be mounted on the backup job.
+ items:
type: string
- passwordKey:
- default: password
- description: passwordKey the map key of the password in
- the connection credential secret
- type: string
- usernameKey:
- default: username
- description: usernameKey the map key of the user in the
- connection credential secret
- type: string
- required:
- - name
- type: object
- required:
- - labelsSelector
- type: object
- required:
- - target
- type: object
- retention:
- description: retention describe how long the Backup should be retained.
- if not set, will be retained forever.
- properties:
- ttl:
- description: ttl is a time string ending with the 'd'|'D'|'h'|'H'
- character to describe how long the Backup should be retained.
- if not set, will be retained forever.
- pattern: ^\d+[d|D|h|H]$
- type: string
- type: object
- schedule:
- description: schedule policy for backup.
+ type: array
+ type: object
+ required:
+ - name
+ type: object
+ type: array
+ backupRepoName:
+ description: backupRepoName is the name of BackupRepo and the backup
+ data will be stored in this repository. If not set, will be stored
+ in the default backup repository.
+ pattern: ^[a-z0-9]([a-z0-9\.\-]*[a-z0-9])?$
+ type: string
+ pathPrefix:
+ description: pathPrefix is the directory inside the backup repository
+ to store the backup content. It is a relative to the path of the
+ backup repository.
+ type: string
+ target:
+ description: target specifies the target information to back up.
properties:
- datafile:
- description: schedule policy for datafile backup.
+ connectionCredential:
+ description: connectionCredential specifies the connection credential
+ to connect to the target database cluster.
properties:
- cronExpression:
- description: the cron expression for schedule, the timezone
- is in UTC. see https://en.wikipedia.org/wiki/Cron.
+ hostKey:
+ description: hostKey specifies the map key of the host in
+ the connection credential secret.
type: string
- enable:
- description: enable or disable the schedule.
- type: boolean
- required:
- - cronExpression
- - enable
- type: object
- logfile:
- description: schedule policy for logfile backup.
- properties:
- cronExpression:
- description: the cron expression for schedule, the timezone
- is in UTC. see https://en.wikipedia.org/wiki/Cron.
+ passwordKey:
+ default: password
+ description: passwordKey specifies the map key of the password
+ in the connection credential secret.
+ type: string
+ portKey:
+ default: port
+ description: portKey specifies the map key of the port in
+ the connection credential secret.
+ type: string
+ secretName:
+ description: secretName refers to the Secret object that contains
+ the connection credential.
+ type: string
+ usernameKey:
+ default: username
+ description: usernameKey specifies the map key of the user
+ in the connection credential secret.
type: string
- enable:
- description: enable or disable the schedule.
- type: boolean
- required:
- - cronExpression
- - enable
type: object
- snapshot:
- description: schedule policy for snapshot backup.
+ podSelector:
+ description: podSelector is used to find the target pod. The volumes
+ of the target pod will be backed up.
properties:
- cronExpression:
- description: the cron expression for schedule, the timezone
- is in UTC. see https://en.wikipedia.org/wiki/Cron.
+ matchExpressions:
+ description: matchExpressions is a list of label selector
+ requirements. The requirements are ANDed.
+ items:
+ description: A label selector requirement is a selector
+ that contains values, a key, and an operator that relates
+ the key and values.
+ properties:
+ key:
+ description: key is the label key that the selector
+ applies to.
+ type: string
+ operator:
+ description: operator represents a key's relationship
+ to a set of values. Valid operators are In, NotIn,
+ Exists and DoesNotExist.
+ type: string
+ values:
+ description: values is an array of string values. If
+ the operator is In or NotIn, the values array must
+ be non-empty. If the operator is Exists or DoesNotExist,
+ the values array must be empty. This array is replaced
+ during a strategic merge patch.
+ items:
+ type: string
+ type: array
+ required:
+ - key
+ - operator
+ type: object
+ type: array
+ matchLabels:
+ additionalProperties:
+ type: string
+ description: matchLabels is a map of {key,value} pairs. A
+ single {key,value} in the matchLabels map is equivalent
+ to an element of matchExpressions, whose key field is "key",
+ the operator is "In", and the values array contains only
+ "value". The requirements are ANDed.
+ type: object
+ strategy:
+ default: Any
+ description: 'strategy specifies the strategy to select the
+ target pod when multiple pods are selected. Valid values
+ are: - All: select all pods that match the labelsSelector.
+ - Any: select any one pod that match the labelsSelector.'
+ enum:
+ - All
+ - Any
type: string
- enable:
- description: enable or disable the schedule.
- type: boolean
- required:
- - cronExpression
- - enable
type: object
- startingDeadlineMinutes:
- description: startingDeadlineMinutes defines the deadline in minutes
- for starting the backup job if it misses scheduled time for
- any reason.
- format: int64
- maximum: 1440
- minimum: 0
- type: integer
- type: object
- snapshot:
- description: the policy for snapshot backup.
- properties:
- backupStatusUpdates:
- description: define how to update metadata for backup status.
- items:
- properties:
- containerName:
- description: which container name that kubectl can execute.
- type: string
- path:
- description: 'specify the json path of backup object for
- patch. example: manifests.backupLog -- means patch the
- backup json path of status.manifests.backupLog.'
- type: string
- script:
- description: the shell Script commands to collect backup
- status metadata. The script must exist in the container
- of ContainerName and the output format must be set to
- JSON. Note that outputting to stderr may cause the result
- format to not be in JSON.
- type: string
- updateStage:
- description: 'when to update the backup status, pre: before
- backup, post: after backup'
- enum:
- - pre
- - post
- type: string
- useTargetPodServiceAccount:
- description: useTargetPodServiceAccount defines whether
- this job requires the service account of the backup target
- pod. if true, will use the service account of the backup
- target pod. otherwise, will use the system service account.
- type: boolean
- required:
- - updateStage
- type: object
- type: array
- backupsHistoryLimit:
- default: 7
- description: the number of automatic backups to retain. Value
- must be non-negative integer. 0 means NO limit on the number
- of backups.
- format: int32
- type: integer
- hooks:
- description: execute hook commands for backup.
+ x-kubernetes-map-type: atomic
+ resources:
+ description: resources specifies the kubernetes resources to back
+ up.
properties:
- containerName:
- description: which container can exec command
- type: string
- image:
- description: exec command with image
- type: string
- postCommands:
- description: post backup to perform commands
+ excluded:
+ description: excluded is a slice of namespaced-scoped resource
+ type names to exclude in the kubernetes resources. The default
+ value is empty.
items:
type: string
type: array
- preCommands:
- description: pre backup to perform commands
+ included:
+ default:
+ - '*'
+ description: included is a slice of namespaced-scoped resource
+ type names to include in the kubernetes resources. The default
+ value is "*", which means all resource types will be included.
items:
type: string
type: array
- type: object
- onFailAttempted:
- description: count of backup stop retries on fail.
- format: int32
- type: integer
- target:
- description: target database cluster for backup.
- properties:
- labelsSelector:
- description: labelsSelector is used to find matching pods.
- Pods that match this label selector are counted to determine
- the number of pods in their corresponding topology domain.
+ selector:
+ description: selector is a metav1.LabelSelector to filter
+ the target kubernetes resources that need to be backed up.
+ If not set, will do not back up any kubernetes resources.
properties:
matchExpressions:
description: matchExpressions is a list of label selector
@@ -628,64 +475,35 @@ spec:
only "value". The requirements are ANDed.
type: object
type: object
- x-kubernetes-preserve-unknown-fields: true
- secret:
- description: secret is used to connect to the target database
- cluster. If not set, secret will be inherited from backup
- policy template. if still not set, the controller will check
- if any system account for dataprotection has been created.
- properties:
- name:
- description: the secret name
- pattern: ^[a-z0-9]([a-z0-9\.\-]*[a-z0-9])?$
- type: string
- passwordKey:
- default: password
- description: passwordKey the map key of the password in
- the connection credential secret
- type: string
- usernameKey:
- default: username
- description: usernameKey the map key of the user in the
- connection credential secret
- type: string
- required:
- - name
- type: object
- required:
- - labelsSelector
+ x-kubernetes-map-type: atomic
type: object
- required:
- - target
+ serviceAccountName:
+ description: serviceAccountName specifies the service account
+ to run the backup workload.
+ type: string
type: object
+ required:
+ - backupMethods
+ - target
type: object
status:
description: BackupPolicyStatus defines the observed state of BackupPolicy
properties:
- failureReason:
- description: the reason if backup policy check failed.
- type: string
- lastScheduleTime:
- description: information when was the last time the job was successfully
- scheduled.
- format: date-time
- type: string
- lastSuccessfulTime:
- description: information when was the last time the job successfully
- completed.
- format: date-time
+ message:
+ description: A human-readable message indicating details about why
+ the BackupPolicy is in this phase.
type: string
observedGeneration:
description: observedGeneration is the most recent generation observed
- for this BackupPolicy. It corresponds to the Cluster's generation,
+ for this BackupPolicy. It refers to the BackupPolicy's generation,
which is updated on mutation by the API Server.
format: int64
type: integer
phase:
- description: 'backup policy phase valid value: Available, Failed.'
+ description: phase - in list of [Available,Unavailable]
enum:
- Available
- - Failed
+ - Unavailable
type: string
type: object
type: object
diff --git a/deploy/helm/crds/dataprotection.kubeblocks.io_backuprepos.yaml b/deploy/helm/crds/dataprotection.kubeblocks.io_backuprepos.yaml
index 4cb9aea608e..00fcc2ab7a4 100644
--- a/deploy/helm/crds/dataprotection.kubeblocks.io_backuprepos.yaml
+++ b/deploy/helm/crds/dataprotection.kubeblocks.io_backuprepos.yaml
@@ -2,8 +2,7 @@ apiVersion: apiextensions.k8s.io/v1
kind: CustomResourceDefinition
metadata:
annotations:
- controller-gen.kubebuilder.io/version: v0.9.0
- creationTimestamp: null
+ controller-gen.kubebuilder.io/version: v0.12.1
labels:
app.kubernetes.io/name: kubeblocks
name: backuprepos.dataprotection.kubeblocks.io
@@ -51,6 +50,13 @@ spec:
spec:
description: BackupRepoSpec defines the desired state of BackupRepo
properties:
+ accessMethod:
+ default: Mount
+ description: Specifies the access method of the backup repo.
+ enum:
+ - Mount
+ - Tool
+ type: string
config:
additionalProperties:
type: string
@@ -69,6 +75,7 @@ spec:
name must be unique.
type: string
type: object
+ x-kubernetes-map-type: atomic
pvReclaimPolicy:
description: The reclaim policy for the PV created by this backup
repo.
@@ -183,6 +190,7 @@ spec:
name must be unique.
type: string
type: object
+ x-kubernetes-map-type: atomic
generatedStorageClassName:
description: generatedStorageClassName indicates the generated storage
class name.
@@ -211,6 +219,10 @@ spec:
description: Backup repo reconciliation phases. Valid values are PreChecking,
Failed, Ready, Deleting.
type: string
+ toolConfigSecretName:
+ description: toolConfigSecretName is the name of the secret containing
+ the configuration for the access tool.
+ type: string
type: object
type: object
served: true
diff --git a/deploy/helm/crds/dataprotection.kubeblocks.io_backups.yaml b/deploy/helm/crds/dataprotection.kubeblocks.io_backups.yaml
index 3bce53724b5..0126b52ced5 100644
--- a/deploy/helm/crds/dataprotection.kubeblocks.io_backups.yaml
+++ b/deploy/helm/crds/dataprotection.kubeblocks.io_backups.yaml
@@ -2,8 +2,7 @@ apiVersion: apiextensions.k8s.io/v1
kind: CustomResourceDefinition
metadata:
annotations:
- controller-gen.kubebuilder.io/version: v0.9.0
- creationTimestamp: null
+ controller-gen.kubebuilder.io/version: v0.12.1
labels:
app.kubernetes.io/name: kubeblocks
name: backups.dataprotection.kubeblocks.io
@@ -19,15 +18,18 @@ spec:
scope: Namespaced
versions:
- additionalPrinterColumns:
- - jsonPath: .spec.backupType
- name: TYPE
+ - jsonPath: .spec.backupPolicyName
+ name: POLICY
+ type: string
+ - jsonPath: .spec.backupMethod
+ name: METHOD
+ type: string
+ - jsonPath: .status.backupRepoName
+ name: REPO
type: string
- jsonPath: .status.phase
name: STATUS
type: string
- - jsonPath: .status.sourceCluster
- name: SOURCE-CLUSTER
- type: string
- jsonPath: .status.totalSize
name: TOTAL-SIZE
type: string
@@ -35,15 +37,18 @@ spec:
name: DURATION
type: string
- jsonPath: .metadata.creationTimestamp
- name: CREATE-TIME
+ name: CREATION-TIME
type: string
- jsonPath: .status.completionTimestamp
name: COMPLETION-TIME
type: string
+ - jsonPath: .status.expiration
+ name: EXPIRATION-TIME
+ type: string
name: v1alpha1
schema:
openAPIV3Schema:
- description: Backup is the Schema for the backups API (defined by User).
+ description: Backup is the Schema for the backups API.
properties:
apiVersion:
description: 'APIVersion defines the versioned schema of this representation
@@ -60,135 +65,444 @@ spec:
spec:
description: BackupSpec defines the desired state of Backup.
properties:
+ backupMethod:
+ description: backupMethod specifies the backup method name that is
+ defined in backupPolicy.
+ type: string
backupPolicyName:
- description: Which backupPolicy is applied to perform this backup
+ description: Which backupPolicy is applied to perform this backup.
pattern: ^[a-z0-9]([a-z0-9\.\-]*[a-z0-9])?$
type: string
- backupType:
- default: datafile
- description: Backup Type. datafile or logfile or snapshot. If not
- set, datafile is the default type.
- enum:
- - datafile
- - logfile
- - snapshot
+ deletionPolicy:
+ allOf:
+ - enum:
+ - Delete
+ - Retain
+ - enum:
+ - Delete
+ - Retain
+ default: Delete
+ description: deletionPolicy determines whether the backup contents
+ stored in backup repository should be deleted when the backup custom
+ resource is deleted. Supported values are "Retain" and "Delete".
+ "Retain" means that the backup content and its physical snapshot
+ on backup repository are kept. "Delete" means that the backup content
+ and its physical snapshot on backup repository are deleted.
type: string
parentBackupName:
- description: if backupType is incremental, parentBackupName is required.
+ description: parentBackupName determines the parent backup name for
+ incremental or differential backup.
+ type: string
+ retentionPeriod:
+ default: 7d
+ description: "retentionPeriod determines a duration up to which the
+ backup should be kept. controller will remove all backups that are
+ older than the RetentionPeriod. For example, RetentionPeriod of
+ `30d` will keep only the backups of last 30 days. Sample duration
+ format: - years: \t2y - months: \t6mo - days: \t\t30d - hours: \t12h
+ - minutes: \t30m You can also combine the above durations. For example:
+ 30d12h30m"
type: string
required:
+ - backupMethod
- backupPolicyName
- - backupType
type: object
status:
description: BackupStatus defines the observed state of Backup.
properties:
- availableReplicas:
- description: availableReplicas available replicas for statefulSet
- which created by backup.
- format: int32
- type: integer
- backupToolName:
- description: backupToolName references the backup tool name.
+ actions:
+ description: actions records the actions information for this backup.
+ items:
+ properties:
+ actionType:
+ description: actionType is the type of the action.
+ type: string
+ availableReplicas:
+ description: availableReplicas available replicas for statefulSet
+ action.
+ format: int32
+ type: integer
+ completionTimestamp:
+ description: completionTimestamp records the time an action
+ was completed.
+ format: date-time
+ type: string
+ failureReason:
+ description: failureReason is an error that caused the backup
+ to fail.
+ type: string
+ name:
+ description: name is the name of the action.
+ type: string
+ objectRef:
+ description: objectRef is the object reference for the action.
+ properties:
+ apiVersion:
+ description: API version of the referent.
+ type: string
+ fieldPath:
+ description: 'If referring to a piece of an object instead
+ of an entire object, this string should contain a valid
+ JSON/Go field access statement, such as desiredState.manifest.containers[2].
+ For example, if the object reference is to a container
+ within a pod, this would take on a value like: "spec.containers{name}"
+ (where "name" refers to the name of the container that
+ triggered the event) or if no container name is specified
+ "spec.containers[2]" (container with index 2 in this pod).
+ This syntax is chosen only to have some well-defined way
+ of referencing a part of an object. TODO: this design
+ is not final and this field is subject to change in the
+ future.'
+ type: string
+ kind:
+ description: 'Kind of the referent. More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#types-kinds'
+ type: string
+ name:
+ description: 'Name of the referent. More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names'
+ type: string
+ namespace:
+ description: 'Namespace of the referent. More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/namespaces/'
+ type: string
+ resourceVersion:
+ description: 'Specific resourceVersion to which this reference
+ is made, if any. More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#concurrency-control-and-consistency'
+ type: string
+ uid:
+ description: 'UID of the referent. More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#uids'
+ type: string
+ type: object
+ x-kubernetes-map-type: atomic
+ phase:
+ description: phase is the current state of the action.
+ type: string
+ startTimestamp:
+ description: startTimestamp records the time an action was started.
+ format: date-time
+ type: string
+ timeRange:
+ description: timeRange records the time range of backed up data,
+ for PITR, this is the time range of recoverable data.
+ properties:
+ end:
+ description: end records the end time of backup.
+ format: date-time
+ type: string
+ start:
+ description: start records the start time of backup.
+ format: date-time
+ type: string
+ type: object
+ totalSize:
+ description: totalSize is the total size of backed up data size.
+ A string with capacity units in the format of "1Gi", "1Mi",
+ "1Ki".
+ type: string
+ volumeSnapshots:
+ description: volumeSnapshots records the volume snapshot status
+ for the action.
+ items:
+ properties:
+ contentName:
+ description: contentName is the name of the volume snapshot
+ content.
+ type: string
+ name:
+ description: name is the name of the volume snapshot.
+ type: string
+ size:
+ description: size is the size of the volume snapshot.
+ type: string
+ volumeName:
+ description: volumeName is the name of the volume.
+ type: string
+ type: object
+ type: array
+ type: object
+ type: array
+ backupMethod:
+ description: backupMethod records the backup method information for
+ this backup. Refer to BackupMethod for more details.
+ properties:
+ actionSetName:
+ description: actionSetName refers to the ActionSet object that
+ defines the backup actions. For volume snapshot backup, the
+ actionSet is not required, the controller will use the CSI volume
+ snapshotter to create the snapshot.
+ type: string
+ env:
+ description: env specifies the environment variables for the backup
+ workload.
+ items:
+ description: EnvVar represents an environment variable present
+ in a Container.
+ properties:
+ name:
+ description: Name of the environment variable. Must be a
+ C_IDENTIFIER.
+ type: string
+ value:
+ description: 'Variable references $(VAR_NAME) are expanded
+ using the previously defined environment variables in
+ the container and any service environment variables. If
+ a variable cannot be resolved, the reference in the input
+ string will be unchanged. Double $$ are reduced to a single
+ $, which allows for escaping the $(VAR_NAME) syntax: i.e.
+ "$$(VAR_NAME)" will produce the string literal "$(VAR_NAME)".
+ Escaped references will never be expanded, regardless
+ of whether the variable exists or not. Defaults to "".'
+ type: string
+ valueFrom:
+ description: Source for the environment variable's value.
+ Cannot be used if value is not empty.
+ properties:
+ configMapKeyRef:
+ description: Selects a key of a ConfigMap.
+ properties:
+ key:
+ description: The key to select.
+ type: string
+ name:
+ description: 'Name of the referent. More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names
+ TODO: Add other useful fields. apiVersion, kind,
+ uid?'
+ type: string
+ optional:
+ description: Specify whether the ConfigMap or its
+ key must be defined
+ type: boolean
+ required:
+ - key
+ type: object
+ x-kubernetes-map-type: atomic
+ fieldRef:
+ description: 'Selects a field of the pod: supports metadata.name,
+ metadata.namespace, `metadata.labels['''']`,
+ `metadata.annotations['''']`, spec.nodeName,
+ spec.serviceAccountName, status.hostIP, status.podIP,
+ status.podIPs.'
+ properties:
+ apiVersion:
+ description: Version of the schema the FieldPath
+ is written in terms of, defaults to "v1".
+ type: string
+ fieldPath:
+ description: Path of the field to select in the
+ specified API version.
+ type: string
+ required:
+ - fieldPath
+ type: object
+ x-kubernetes-map-type: atomic
+ resourceFieldRef:
+ description: 'Selects a resource of the container: only
+ resources limits and requests (limits.cpu, limits.memory,
+ limits.ephemeral-storage, requests.cpu, requests.memory
+ and requests.ephemeral-storage) are currently supported.'
+ properties:
+ containerName:
+ description: 'Container name: required for volumes,
+ optional for env vars'
+ type: string
+ divisor:
+ anyOf:
+ - type: integer
+ - type: string
+ description: Specifies the output format of the
+ exposed resources, defaults to "1"
+ pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$
+ x-kubernetes-int-or-string: true
+ resource:
+ description: 'Required: resource to select'
+ type: string
+ required:
+ - resource
+ type: object
+ x-kubernetes-map-type: atomic
+ secretKeyRef:
+ description: Selects a key of a secret in the pod's
+ namespace
+ properties:
+ key:
+ description: The key of the secret to select from. Must
+ be a valid secret key.
+ type: string
+ name:
+ description: 'Name of the referent. More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names
+ TODO: Add other useful fields. apiVersion, kind,
+ uid?'
+ type: string
+ optional:
+ description: Specify whether the Secret or its key
+ must be defined
+ type: boolean
+ required:
+ - key
+ type: object
+ x-kubernetes-map-type: atomic
+ type: object
+ required:
+ - name
+ type: object
+ type: array
+ name:
+ description: the name of backup method.
+ pattern: ^[a-z0-9]([a-z0-9\.\-]*[a-z0-9])?$
+ type: string
+ runtimeSettings:
+ description: runtimeSettings specifies runtime settings for the
+ backup workload container.
+ properties:
+ resources:
+ description: 'resources specifies the resource required by
+ container. More info: https://kubernetes.io/docs/concepts/configuration/manage-compute-resources-container/'
+ properties:
+ claims:
+ description: "Claims lists the names of resources, defined
+ in spec.resourceClaims, that are used by this container.
+ \n This is an alpha field and requires enabling the
+ DynamicResourceAllocation feature gate. \n This field
+ is immutable. It can only be set for containers."
+ items:
+ description: ResourceClaim references one entry in PodSpec.ResourceClaims.
+ properties:
+ name:
+ description: Name must match the name of one entry
+ in pod.spec.resourceClaims of the Pod where this
+ field is used. It makes that resource available
+ inside a container.
+ type: string
+ required:
+ - name
+ type: object
+ type: array
+ x-kubernetes-list-map-keys:
+ - name
+ x-kubernetes-list-type: map
+ limits:
+ additionalProperties:
+ anyOf:
+ - type: integer
+ - type: string
+ pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$
+ x-kubernetes-int-or-string: true
+ description: 'Limits describes the maximum amount of compute
+ resources allowed. More info: https://kubernetes.io/docs/concepts/configuration/manage-resources-containers/'
+ type: object
+ requests:
+ additionalProperties:
+ anyOf:
+ - type: integer
+ - type: string
+ pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$
+ x-kubernetes-int-or-string: true
+ description: 'Requests describes the minimum amount of
+ compute resources required. If Requests is omitted for
+ a container, it defaults to Limits if that is explicitly
+ specified, otherwise to an implementation-defined value.
+ Requests cannot exceed Limits. More info: https://kubernetes.io/docs/concepts/configuration/manage-resources-containers/'
+ type: object
+ type: object
+ type: object
+ snapshotVolumes:
+ default: false
+ description: snapshotVolumes specifies whether to take snapshots
+ of persistent volumes. if true, the BackupScript is not required,
+ the controller will use the CSI volume snapshotter to create
+ the snapshot.
+ type: boolean
+ targetVolumes:
+ description: targetVolumes specifies which volumes from the target
+ should be mounted in the backup workload.
+ properties:
+ volumeMounts:
+ description: volumeMounts specifies the mount for the volumes
+ specified in `Volumes` section.
+ items:
+ description: VolumeMount describes a mounting of a Volume
+ within a container.
+ properties:
+ mountPath:
+ description: Path within the container at which the
+ volume should be mounted. Must not contain ':'.
+ type: string
+ mountPropagation:
+ description: mountPropagation determines how mounts
+ are propagated from the host to container and the
+ other way around. When not set, MountPropagationNone
+ is used. This field is beta in 1.10.
+ type: string
+ name:
+ description: This must match the Name of a Volume.
+ type: string
+ readOnly:
+ description: Mounted read-only if true, read-write otherwise
+ (false or unspecified). Defaults to false.
+ type: boolean
+ subPath:
+ description: Path within the volume from which the container's
+ volume should be mounted. Defaults to "" (volume's
+ root).
+ type: string
+ subPathExpr:
+ description: Expanded path within the volume from which
+ the container's volume should be mounted. Behaves
+ similarly to SubPath but environment variable references
+ $(VAR_NAME) are expanded using the container's environment.
+ Defaults to "" (volume's root). SubPathExpr and SubPath
+ are mutually exclusive.
+ type: string
+ required:
+ - mountPath
+ - name
+ type: object
+ type: array
+ volumes:
+ description: Volumes indicates the list of volumes of targeted
+ application that should be mounted on the backup job.
+ items:
+ type: string
+ type: array
+ type: object
+ required:
+ - name
+ type: object
+ backupRepoName:
+ description: backupRepoName is the name of the backup repository.
type: string
completionTimestamp:
- description: Date/time when the backup finished being processed.
+ description: completionTimestamp records the time a backup was completed.
+ Completion time is recorded even on failed backups. The server's
+ time is used for CompletionTimestamp.
format: date-time
type: string
duration:
description: The duration time of backup execution. When converted
- to a string, the form is "1h2m0.5s".
+ to a string, the format is "1h2m0.5s".
type: string
expiration:
- description: The date and time when the Backup is eligible for garbage
- collection. 'null' means the Backup is NOT be cleaned except delete
+ description: expiration is when this backup is eligible for garbage
+ collection. 'null' means the Backup will NOT be cleaned except delete
manual.
format: date-time
type: string
failureReason:
- description: The reason for a backup failure.
+ description: failureReason is an error that caused the backup to fail.
type: string
- logFilePersistentVolumeClaimName:
- description: logFilePersistentVolumeClaimName saves the logfile backup
- data.
+ formatVersion:
+ description: formatVersion is the backup format version, including
+ major, minor and patch version.
type: string
- manifests:
- description: manifests determines the backup metadata info.
- properties:
- backupLog:
- description: backupLog records startTime and stopTime of data
- logging.
- properties:
- startTime:
- description: startTime records the start time of data logging.
- format: date-time
- type: string
- stopTime:
- description: stopTime records the stop time of data logging.
- format: date-time
- type: string
- type: object
- backupSnapshot:
- description: snapshot records the volume snapshot metadata.
- properties:
- volumeSnapshotContentName:
- description: volumeSnapshotContentName specifies the name
- of a pre-existing VolumeSnapshotContent object representing
- an existing volume snapshot. This field should be set if
- the snapshot already exists and only needs a representation
- in Kubernetes. This field is immutable.
- type: string
- volumeSnapshotName:
- description: volumeSnapshotName records the volumeSnapshot
- name.
- type: string
- type: object
- backupTool:
- description: backupTool records information about backup files
- generated by the backup tool.
- properties:
- checkpoint:
- description: backup checkpoint, for incremental backup.
- type: string
- checksum:
- description: checksum of backup file, generated by md5 or
- sha1 or sha256.
- type: string
- filePath:
- description: filePath records the file path of backup.
- type: string
- logFilePath:
- description: logFilePath records the log file path of backup.
- type: string
- uploadTotalSize:
- description: Backup upload total size. A string with capacity
- units in the form of "1Gi", "1Mi", "1Ki".
- type: string
- volumeName:
- description: volumeName records volume name of backup data
- pvc.
- type: string
- type: object
- target:
- description: target records the target cluster metadata string,
- which is in JSON format.
- type: string
- userContext:
- additionalProperties:
- type: string
- description: userContext stores some loosely structured and extensible
- information.
- type: object
- type: object
- parentBackupName:
- description: Records parentBackupName if backupType is incremental.
+ path:
+ description: path is the directory inside the backup repository where
+ the backup data is stored. It is an absolute path in the backup
+ repository.
type: string
persistentVolumeClaimName:
- description: remoteVolume saves the backup data.
+ description: persistentVolumeClaimName is the name of the persistent
+ volume claim that is used to store the backup data.
type: string
phase:
- description: BackupPhase The current phase. Valid values are New,
- InProgress, Completed, Failed.
+ description: phase is the current state of the Backup.
enum:
- New
- InProgress
@@ -197,18 +511,209 @@ spec:
- Failed
- Deleting
type: string
- sourceCluster:
- description: sourceCluster records the source cluster information
- for this backup.
- type: string
startTimestamp:
- description: Date/time when the backup started being processed.
+ description: startTimestamp records the time a backup was started.
+ The server's time is used for StartTimestamp.
format: date-time
type: string
+ target:
+ description: target records the target information for this backup.
+ properties:
+ connectionCredential:
+ description: connectionCredential specifies the connection credential
+ to connect to the target database cluster.
+ properties:
+ hostKey:
+ description: hostKey specifies the map key of the host in
+ the connection credential secret.
+ type: string
+ passwordKey:
+ default: password
+ description: passwordKey specifies the map key of the password
+ in the connection credential secret.
+ type: string
+ portKey:
+ default: port
+ description: portKey specifies the map key of the port in
+ the connection credential secret.
+ type: string
+ secretName:
+ description: secretName refers to the Secret object that contains
+ the connection credential.
+ type: string
+ usernameKey:
+ default: username
+ description: usernameKey specifies the map key of the user
+ in the connection credential secret.
+ type: string
+ type: object
+ podSelector:
+ description: podSelector is used to find the target pod. The volumes
+ of the target pod will be backed up.
+ properties:
+ matchExpressions:
+ description: matchExpressions is a list of label selector
+ requirements. The requirements are ANDed.
+ items:
+ description: A label selector requirement is a selector
+ that contains values, a key, and an operator that relates
+ the key and values.
+ properties:
+ key:
+ description: key is the label key that the selector
+ applies to.
+ type: string
+ operator:
+ description: operator represents a key's relationship
+ to a set of values. Valid operators are In, NotIn,
+ Exists and DoesNotExist.
+ type: string
+ values:
+ description: values is an array of string values. If
+ the operator is In or NotIn, the values array must
+ be non-empty. If the operator is Exists or DoesNotExist,
+ the values array must be empty. This array is replaced
+ during a strategic merge patch.
+ items:
+ type: string
+ type: array
+ required:
+ - key
+ - operator
+ type: object
+ type: array
+ matchLabels:
+ additionalProperties:
+ type: string
+ description: matchLabels is a map of {key,value} pairs. A
+ single {key,value} in the matchLabels map is equivalent
+ to an element of matchExpressions, whose key field is "key",
+ the operator is "In", and the values array contains only
+ "value". The requirements are ANDed.
+ type: object
+ strategy:
+ default: Any
+ description: 'strategy specifies the strategy to select the
+ target pod when multiple pods are selected. Valid values
+ are: - All: select all pods that match the labelsSelector.
+ - Any: select any one pod that match the labelsSelector.'
+ enum:
+ - All
+ - Any
+ type: string
+ type: object
+ x-kubernetes-map-type: atomic
+ resources:
+ description: resources specifies the kubernetes resources to back
+ up.
+ properties:
+ excluded:
+ description: excluded is a slice of namespaced-scoped resource
+ type names to exclude in the kubernetes resources. The default
+ value is empty.
+ items:
+ type: string
+ type: array
+ included:
+ default:
+ - '*'
+ description: included is a slice of namespaced-scoped resource
+ type names to include in the kubernetes resources. The default
+ value is "*", which means all resource types will be included.
+ items:
+ type: string
+ type: array
+ selector:
+ description: selector is a metav1.LabelSelector to filter
+ the target kubernetes resources that need to be backed up.
+ If not set, will do not back up any kubernetes resources.
+ properties:
+ matchExpressions:
+ description: matchExpressions is a list of label selector
+ requirements. The requirements are ANDed.
+ items:
+ description: A label selector requirement is a selector
+ that contains values, a key, and an operator that
+ relates the key and values.
+ properties:
+ key:
+ description: key is the label key that the selector
+ applies to.
+ type: string
+ operator:
+ description: operator represents a key's relationship
+ to a set of values. Valid operators are In, NotIn,
+ Exists and DoesNotExist.
+ type: string
+ values:
+ description: values is an array of string values.
+ If the operator is In or NotIn, the values array
+ must be non-empty. If the operator is Exists or
+ DoesNotExist, the values array must be empty.
+ This array is replaced during a strategic merge
+ patch.
+ items:
+ type: string
+ type: array
+ required:
+ - key
+ - operator
+ type: object
+ type: array
+ matchLabels:
+ additionalProperties:
+ type: string
+ description: matchLabels is a map of {key,value} pairs.
+ A single {key,value} in the matchLabels map is equivalent
+ to an element of matchExpressions, whose key field is
+ "key", the operator is "In", and the values array contains
+ only "value". The requirements are ANDed.
+ type: object
+ type: object
+ x-kubernetes-map-type: atomic
+ type: object
+ serviceAccountName:
+ description: serviceAccountName specifies the service account
+ to run the backup workload.
+ type: string
+ type: object
+ timeRange:
+ description: timeRange records the time range of backed up data, for
+ PITR, this is the time range of recoverable data.
+ properties:
+ end:
+ description: end records the end time of backup.
+ format: date-time
+ type: string
+ start:
+ description: start records the start time of backup.
+ format: date-time
+ type: string
+ type: object
totalSize:
- description: Backup total size. A string with capacity units in the
- form of "1Gi", "1Mi", "1Ki".
+ description: totalSize is the total size of backed up data size. A
+ string with capacity units in the format of "1Gi", "1Mi", "1Ki".
type: string
+ volumeSnapshots:
+ description: volumeSnapshots records the volume snapshot status for
+ the action.
+ items:
+ properties:
+ contentName:
+ description: contentName is the name of the volume snapshot
+ content.
+ type: string
+ name:
+ description: name is the name of the volume snapshot.
+ type: string
+ size:
+ description: size is the size of the volume snapshot.
+ type: string
+ volumeName:
+ description: volumeName is the name of the volume.
+ type: string
+ type: object
+ type: array
type: object
type: object
served: true
diff --git a/deploy/helm/crds/dataprotection.kubeblocks.io_backupschedules.yaml b/deploy/helm/crds/dataprotection.kubeblocks.io_backupschedules.yaml
new file mode 100644
index 00000000000..40d07aa5fc7
--- /dev/null
+++ b/deploy/helm/crds/dataprotection.kubeblocks.io_backupschedules.yaml
@@ -0,0 +1,141 @@
+apiVersion: apiextensions.k8s.io/v1
+kind: CustomResourceDefinition
+metadata:
+ annotations:
+ controller-gen.kubebuilder.io/version: v0.12.1
+ labels:
+ app.kubernetes.io/name: kubeblocks
+ name: backupschedules.dataprotection.kubeblocks.io
+spec:
+ group: dataprotection.kubeblocks.io
+ names:
+ categories:
+ - kubeblocks
+ kind: BackupSchedule
+ listKind: BackupScheduleList
+ plural: backupschedules
+ shortNames:
+ - bs
+ singular: backupschedule
+ scope: Namespaced
+ versions:
+ - additionalPrinterColumns:
+ - jsonPath: .status.phase
+ name: STATUS
+ type: string
+ - jsonPath: .metadata.creationTimestamp
+ name: AGE
+ type: date
+ name: v1alpha1
+ schema:
+ openAPIV3Schema:
+ description: BackupSchedule is the Schema for the backupschedules API.
+ properties:
+ apiVersion:
+ description: 'APIVersion defines the versioned schema of this representation
+ of an object. Servers should convert recognized schemas to the latest
+ internal value, and may reject unrecognized values. More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#resources'
+ type: string
+ kind:
+ description: 'Kind is a string value representing the REST resource this
+ object represents. Servers may infer this from the endpoint the client
+ submits requests to. Cannot be updated. In CamelCase. More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#types-kinds'
+ type: string
+ metadata:
+ type: object
+ spec:
+ description: BackupScheduleSpec defines the desired state of BackupSchedule.
+ properties:
+ backupPolicyName:
+ description: Which backupPolicy is applied to perform this backup.
+ pattern: ^[a-z0-9]([a-z0-9\.\-]*[a-z0-9])?$
+ type: string
+ schedules:
+ description: schedules defines the list of backup schedules.
+ items:
+ properties:
+ backupMethod:
+ description: backupMethod specifies the backup method name that
+ is defined in backupPolicy.
+ type: string
+ cronExpression:
+ description: the cron expression for schedule, the timezone
+ is in UTC. see https://en.wikipedia.org/wiki/Cron.
+ type: string
+ enabled:
+ description: enabled specifies whether the backup schedule is
+ enabled or not.
+ type: boolean
+ retentionPeriod:
+ default: 7d
+ description: "retentionPeriod determines a duration up to which
+ the backup should be kept. controller will remove all backups
+ that are older than the RetentionPeriod. For example, RetentionPeriod
+ of `30d` will keep only the backups of last 30 days. Sample
+ duration format: - years: \t2y - months: \t6mo - days: \t\t30d
+ - hours: \t12h - minutes: \t30m You can also combine the above
+ durations. For example: 30d12h30m"
+ type: string
+ required:
+ - backupMethod
+ - cronExpression
+ type: object
+ minItems: 1
+ type: array
+ startingDeadlineMinutes:
+ description: startingDeadlineMinutes defines the deadline in minutes
+ for starting the backup workload if it misses scheduled time for
+ any reason.
+ format: int64
+ maximum: 1440
+ minimum: 0
+ type: integer
+ required:
+ - backupPolicyName
+ - schedules
+ type: object
+ status:
+ description: BackupScheduleStatus defines the observed state of BackupSchedule.
+ properties:
+ failureReason:
+ description: failureReason is an error that caused the backup to fail.
+ type: string
+ observedGeneration:
+ description: observedGeneration is the most recent generation observed
+ for this BackupSchedule. It refers to the BackupSchedule's generation,
+ which is updated on mutation by the API Server.
+ format: int64
+ type: integer
+ phase:
+ description: phase describes the phase of the BackupSchedule.
+ type: string
+ schedules:
+ additionalProperties:
+ description: ScheduleStatus defines the status of each schedule.
+ properties:
+ failureReason:
+ description: failureReason is an error that caused the backup
+ to fail.
+ type: string
+ lastScheduleTime:
+ description: lastScheduleTime records the last time the backup
+ was scheduled.
+ format: date-time
+ type: string
+ lastSuccessfulTime:
+ description: lastSuccessfulTime records the last time the backup
+ was successfully completed.
+ format: date-time
+ type: string
+ phase:
+ description: phase describes the phase of the schedule.
+ type: string
+ type: object
+ description: schedules describes the status of each schedule.
+ type: object
+ type: object
+ type: object
+ served: true
+ storage: true
+ subresources:
+ status: {}
diff --git a/deploy/helm/crds/dataprotection.kubeblocks.io_backuptools.yaml b/deploy/helm/crds/dataprotection.kubeblocks.io_backuptools.yaml
deleted file mode 100644
index 2bec3e71768..00000000000
--- a/deploy/helm/crds/dataprotection.kubeblocks.io_backuptools.yaml
+++ /dev/null
@@ -1,324 +0,0 @@
-apiVersion: apiextensions.k8s.io/v1
-kind: CustomResourceDefinition
-metadata:
- annotations:
- controller-gen.kubebuilder.io/version: v0.9.0
- creationTimestamp: null
- labels:
- app.kubernetes.io/name: kubeblocks
- name: backuptools.dataprotection.kubeblocks.io
-spec:
- group: dataprotection.kubeblocks.io
- names:
- categories:
- - kubeblocks
- kind: BackupTool
- listKind: BackupToolList
- plural: backuptools
- singular: backuptool
- scope: Cluster
- versions:
- - name: v1alpha1
- schema:
- openAPIV3Schema:
- description: BackupTool is the Schema for the backuptools API (defined by
- provider)
- properties:
- apiVersion:
- description: 'APIVersion defines the versioned schema of this representation
- of an object. Servers should convert recognized schemas to the latest
- internal value, and may reject unrecognized values. More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#resources'
- type: string
- kind:
- description: 'Kind is a string value representing the REST resource this
- object represents. Servers may infer this from the endpoint the client
- submits requests to. Cannot be updated. In CamelCase. More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#types-kinds'
- type: string
- metadata:
- type: object
- spec:
- description: BackupToolSpec defines the desired state of BackupTool
- properties:
- backupCommands:
- description: Array of command that apps can do database backup. from
- invoke args the order of commands follows the order of array.
- items:
- type: string
- type: array
- deployKind:
- default: job
- description: 'which kind for run a backup tool, supported values:
- job, statefulSet.'
- enum:
- - job
- - statefulSet
- type: string
- env:
- description: List of environment variables to set in the container.
- items:
- description: EnvVar represents an environment variable present in
- a Container.
- properties:
- name:
- description: Name of the environment variable. Must be a C_IDENTIFIER.
- type: string
- value:
- description: 'Variable references $(VAR_NAME) are expanded using
- the previously defined environment variables in the container
- and any service environment variables. If a variable cannot
- be resolved, the reference in the input string will be unchanged.
- Double $$ are reduced to a single $, which allows for escaping
- the $(VAR_NAME) syntax: i.e. "$$(VAR_NAME)" will produce the
- string literal "$(VAR_NAME)". Escaped references will never
- be expanded, regardless of whether the variable exists or
- not. Defaults to "".'
- type: string
- valueFrom:
- description: Source for the environment variable's value. Cannot
- be used if value is not empty.
- properties:
- configMapKeyRef:
- description: Selects a key of a ConfigMap.
- properties:
- key:
- description: The key to select.
- type: string
- name:
- description: 'Name of the referent. More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names
- TODO: Add other useful fields. apiVersion, kind, uid?'
- type: string
- optional:
- description: Specify whether the ConfigMap or its key
- must be defined
- type: boolean
- required:
- - key
- type: object
- fieldRef:
- description: 'Selects a field of the pod: supports metadata.name,
- metadata.namespace, `metadata.labels['''']`, `metadata.annotations['''']`,
- spec.nodeName, spec.serviceAccountName, status.hostIP,
- status.podIP, status.podIPs.'
- properties:
- apiVersion:
- description: Version of the schema the FieldPath is
- written in terms of, defaults to "v1".
- type: string
- fieldPath:
- description: Path of the field to select in the specified
- API version.
- type: string
- required:
- - fieldPath
- type: object
- resourceFieldRef:
- description: 'Selects a resource of the container: only
- resources limits and requests (limits.cpu, limits.memory,
- limits.ephemeral-storage, requests.cpu, requests.memory
- and requests.ephemeral-storage) are currently supported.'
- properties:
- containerName:
- description: 'Container name: required for volumes,
- optional for env vars'
- type: string
- divisor:
- anyOf:
- - type: integer
- - type: string
- description: Specifies the output format of the exposed
- resources, defaults to "1"
- pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$
- x-kubernetes-int-or-string: true
- resource:
- description: 'Required: resource to select'
- type: string
- required:
- - resource
- type: object
- secretKeyRef:
- description: Selects a key of a secret in the pod's namespace
- properties:
- key:
- description: The key of the secret to select from. Must
- be a valid secret key.
- type: string
- name:
- description: 'Name of the referent. More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names
- TODO: Add other useful fields. apiVersion, kind, uid?'
- type: string
- optional:
- description: Specify whether the Secret or its key must
- be defined
- type: boolean
- required:
- - key
- type: object
- type: object
- required:
- - name
- type: object
- type: array
- x-kubernetes-preserve-unknown-fields: true
- envFrom:
- description: List of sources to populate environment variables in
- the container. The keys defined within a source must be a C_IDENTIFIER.
- All invalid keys will be reported as an event when the container
- is starting. When a key exists in multiple sources, the value associated
- with the last source will take precedence. Values defined by an
- Env with a duplicate key will take precedence. Cannot be updated.
- items:
- description: EnvFromSource represents the source of a set of ConfigMaps
- properties:
- configMapRef:
- description: The ConfigMap to select from
- properties:
- name:
- description: 'Name of the referent. More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names
- TODO: Add other useful fields. apiVersion, kind, uid?'
- type: string
- optional:
- description: Specify whether the ConfigMap must be defined
- type: boolean
- type: object
- prefix:
- description: An optional identifier to prepend to each key in
- the ConfigMap. Must be a C_IDENTIFIER.
- type: string
- secretRef:
- description: The Secret to select from
- properties:
- name:
- description: 'Name of the referent. More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names
- TODO: Add other useful fields. apiVersion, kind, uid?'
- type: string
- optional:
- description: Specify whether the Secret must be defined
- type: boolean
- type: object
- type: object
- type: array
- x-kubernetes-preserve-unknown-fields: true
- image:
- description: Backup tool Container image name.
- type: string
- incrementalBackupCommands:
- description: Array of command that apps can do database incremental
- backup. like xtrabackup, that can performs an incremental backup
- file.
- items:
- type: string
- type: array
- logical:
- description: backup tool can support logical restore, in this case,
- restore NOT RESTART database.
- properties:
- incrementalRestoreCommands:
- description: Array of incremental restore commands.
- items:
- type: string
- type: array
- podScope:
- default: All
- description: 'podScope defines the pod scope for restore from
- backup, supported values: - ''All'' will exec the restore command
- on all pods. - ''ReadWrite'' will pick a ReadWrite pod to exec
- the restore command.'
- enum:
- - All
- - ReadWrite
- type: string
- restoreCommands:
- description: Array of command that apps can perform database restore.
- like xtrabackup, that can performs restore mysql from files.
- items:
- type: string
- type: array
- type: object
- physical:
- description: backup tool can support physical restore, in this case,
- restore must be RESTART database.
- properties:
- incrementalRestoreCommands:
- description: Array of incremental restore commands.
- items:
- type: string
- type: array
- relyOnLogfile:
- description: relyOnLogfile defines whether the current recovery
- relies on log files
- type: boolean
- restoreCommands:
- description: Array of command that apps can perform database restore.
- like xtrabackup, that can performs restore mysql from files.
- items:
- type: string
- type: array
- type: object
- resources:
- description: Compute Resources required by this container. Cannot
- be updated.
- properties:
- claims:
- description: "Claims lists the names of resources, defined in
- spec.resourceClaims, that are used by this container. \n This
- is an alpha field and requires enabling the DynamicResourceAllocation
- feature gate. \n This field is immutable. It can only be set
- for containers."
- items:
- description: ResourceClaim references one entry in PodSpec.ResourceClaims.
- properties:
- name:
- description: Name must match the name of one entry in pod.spec.resourceClaims
- of the Pod where this field is used. It makes that resource
- available inside a container.
- type: string
- required:
- - name
- type: object
- type: array
- x-kubernetes-list-map-keys:
- - name
- x-kubernetes-list-type: map
- limits:
- additionalProperties:
- anyOf:
- - type: integer
- - type: string
- pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$
- x-kubernetes-int-or-string: true
- description: 'Limits describes the maximum amount of compute resources
- allowed. More info: https://kubernetes.io/docs/concepts/configuration/manage-resources-containers/'
- type: object
- requests:
- additionalProperties:
- anyOf:
- - type: integer
- - type: string
- pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$
- x-kubernetes-int-or-string: true
- description: 'Requests describes the minimum amount of compute
- resources required. If Requests is omitted for a container,
- it defaults to Limits if that is explicitly specified, otherwise
- to an implementation-defined value. More info: https://kubernetes.io/docs/concepts/configuration/manage-resources-containers/'
- type: object
- type: object
- x-kubernetes-preserve-unknown-fields: true
- type:
- default: file
- description: the type of backup tool, file or pitr
- enum:
- - file
- - pitr
- type: string
- required:
- - backupCommands
- - image
- type: object
- status:
- description: BackupToolStatus defines the observed state of BackupTool
- type: object
- type: object
- served: true
- storage: true
- subresources:
- status: {}
diff --git a/deploy/helm/crds/dataprotection.kubeblocks.io_restorejobs.yaml b/deploy/helm/crds/dataprotection.kubeblocks.io_restorejobs.yaml
deleted file mode 100644
index 917f6a6390f..00000000000
--- a/deploy/helm/crds/dataprotection.kubeblocks.io_restorejobs.yaml
+++ /dev/null
@@ -1,1778 +0,0 @@
-apiVersion: apiextensions.k8s.io/v1
-kind: CustomResourceDefinition
-metadata:
- annotations:
- controller-gen.kubebuilder.io/version: v0.9.0
- creationTimestamp: null
- labels:
- app.kubernetes.io/name: kubeblocks
- name: restorejobs.dataprotection.kubeblocks.io
-spec:
- group: dataprotection.kubeblocks.io
- names:
- categories:
- - kubeblocks
- kind: RestoreJob
- listKind: RestoreJobList
- plural: restorejobs
- singular: restorejob
- scope: Namespaced
- versions:
- - additionalPrinterColumns:
- - jsonPath: .status.phase
- name: STATUS
- type: string
- - jsonPath: .status.completionTimestamp
- name: COMPLETION-TIME
- type: date
- - jsonPath: .metadata.creationTimestamp
- name: AGE
- type: date
- name: v1alpha1
- schema:
- openAPIV3Schema:
- description: RestoreJob is the Schema for the restorejobs API (defined by
- User)
- properties:
- apiVersion:
- description: 'APIVersion defines the versioned schema of this representation
- of an object. Servers should convert recognized schemas to the latest
- internal value, and may reject unrecognized values. More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#resources'
- type: string
- kind:
- description: 'Kind is a string value representing the REST resource this
- object represents. Servers may infer this from the endpoint the client
- submits requests to. Cannot be updated. In CamelCase. More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#types-kinds'
- type: string
- metadata:
- type: object
- spec:
- description: RestoreJobSpec defines the desired state of RestoreJob
- properties:
- backupJobName:
- description: Specified one backupJob to restore.
- type: string
- onFailAttempted:
- description: count of backup stop retries on fail.
- format: int32
- type: integer
- target:
- description: the target database workload to restore
- properties:
- labelsSelector:
- description: labelsSelector is used to find matching pods. Pods
- that match this label selector are counted to determine the
- number of pods in their corresponding topology domain.
- properties:
- matchExpressions:
- description: matchExpressions is a list of label selector
- requirements. The requirements are ANDed.
- items:
- description: A label selector requirement is a selector
- that contains values, a key, and an operator that relates
- the key and values.
- properties:
- key:
- description: key is the label key that the selector
- applies to.
- type: string
- operator:
- description: operator represents a key's relationship
- to a set of values. Valid operators are In, NotIn,
- Exists and DoesNotExist.
- type: string
- values:
- description: values is an array of string values. If
- the operator is In or NotIn, the values array must
- be non-empty. If the operator is Exists or DoesNotExist,
- the values array must be empty. This array is replaced
- during a strategic merge patch.
- items:
- type: string
- type: array
- required:
- - key
- - operator
- type: object
- type: array
- matchLabels:
- additionalProperties:
- type: string
- description: matchLabels is a map of {key,value} pairs. A
- single {key,value} in the matchLabels map is equivalent
- to an element of matchExpressions, whose key field is "key",
- the operator is "In", and the values array contains only
- "value". The requirements are ANDed.
- type: object
- type: object
- x-kubernetes-preserve-unknown-fields: true
- secret:
- description: secret is used to connect to the target database
- cluster. If not set, secret will be inherited from backup policy
- template. if still not set, the controller will check if any
- system account for dataprotection has been created.
- properties:
- name:
- description: the secret name
- pattern: ^[a-z0-9]([a-z0-9\.\-]*[a-z0-9])?$
- type: string
- passwordKey:
- default: password
- description: passwordKey the map key of the password in the
- connection credential secret
- type: string
- usernameKey:
- default: username
- description: usernameKey the map key of the user in the connection
- credential secret
- type: string
- required:
- - name
- type: object
- required:
- - labelsSelector
- type: object
- targetVolumeMounts:
- description: array of restore volume mounts .
- items:
- description: VolumeMount describes a mounting of a Volume within
- a container.
- properties:
- mountPath:
- description: Path within the container at which the volume should
- be mounted. Must not contain ':'.
- type: string
- mountPropagation:
- description: mountPropagation determines how mounts are propagated
- from the host to container and the other way around. When
- not set, MountPropagationNone is used. This field is beta
- in 1.10.
- type: string
- name:
- description: This must match the Name of a Volume.
- type: string
- readOnly:
- description: Mounted read-only if true, read-write otherwise
- (false or unspecified). Defaults to false.
- type: boolean
- subPath:
- description: Path within the volume from which the container's
- volume should be mounted. Defaults to "" (volume's root).
- type: string
- subPathExpr:
- description: Expanded path within the volume from which the
- container's volume should be mounted. Behaves similarly to
- SubPath but environment variable references $(VAR_NAME) are
- expanded using the container's environment. Defaults to ""
- (volume's root). SubPathExpr and SubPath are mutually exclusive.
- type: string
- required:
- - mountPath
- - name
- type: object
- minItems: 1
- type: array
- x-kubernetes-preserve-unknown-fields: true
- targetVolumes:
- description: array of restore volumes .
- items:
- description: Volume represents a named volume in a pod that may
- be accessed by any container in the pod.
- properties:
- awsElasticBlockStore:
- description: 'awsElasticBlockStore represents an AWS Disk resource
- that is attached to a kubelet''s host machine and then exposed
- to the pod. More info: https://kubernetes.io/docs/concepts/storage/volumes#awselasticblockstore'
- properties:
- fsType:
- description: 'fsType is the filesystem type of the volume
- that you want to mount. Tip: Ensure that the filesystem
- type is supported by the host operating system. Examples:
- "ext4", "xfs", "ntfs". Implicitly inferred to be "ext4"
- if unspecified. More info: https://kubernetes.io/docs/concepts/storage/volumes#awselasticblockstore
- TODO: how do we prevent errors in the filesystem from
- compromising the machine'
- type: string
- partition:
- description: 'partition is the partition in the volume that
- you want to mount. If omitted, the default is to mount
- by volume name. Examples: For volume /dev/sda1, you specify
- the partition as "1". Similarly, the volume partition
- for /dev/sda is "0" (or you can leave the property empty).'
- format: int32
- type: integer
- readOnly:
- description: 'readOnly value true will force the readOnly
- setting in VolumeMounts. More info: https://kubernetes.io/docs/concepts/storage/volumes#awselasticblockstore'
- type: boolean
- volumeID:
- description: 'volumeID is unique ID of the persistent disk
- resource in AWS (Amazon EBS volume). More info: https://kubernetes.io/docs/concepts/storage/volumes#awselasticblockstore'
- type: string
- required:
- - volumeID
- type: object
- azureDisk:
- description: azureDisk represents an Azure Data Disk mount on
- the host and bind mount to the pod.
- properties:
- cachingMode:
- description: 'cachingMode is the Host Caching mode: None,
- Read Only, Read Write.'
- type: string
- diskName:
- description: diskName is the Name of the data disk in the
- blob storage
- type: string
- diskURI:
- description: diskURI is the URI of data disk in the blob
- storage
- type: string
- fsType:
- description: fsType is Filesystem type to mount. Must be
- a filesystem type supported by the host operating system.
- Ex. "ext4", "xfs", "ntfs". Implicitly inferred to be "ext4"
- if unspecified.
- type: string
- kind:
- description: 'kind expected values are Shared: multiple
- blob disks per storage account Dedicated: single blob
- disk per storage account Managed: azure managed data
- disk (only in managed availability set). defaults to shared'
- type: string
- readOnly:
- description: readOnly Defaults to false (read/write). ReadOnly
- here will force the ReadOnly setting in VolumeMounts.
- type: boolean
- required:
- - diskName
- - diskURI
- type: object
- azureFile:
- description: azureFile represents an Azure File Service mount
- on the host and bind mount to the pod.
- properties:
- readOnly:
- description: readOnly defaults to false (read/write). ReadOnly
- here will force the ReadOnly setting in VolumeMounts.
- type: boolean
- secretName:
- description: secretName is the name of secret that contains
- Azure Storage Account Name and Key
- type: string
- shareName:
- description: shareName is the azure share Name
- type: string
- required:
- - secretName
- - shareName
- type: object
- cephfs:
- description: cephFS represents a Ceph FS mount on the host that
- shares a pod's lifetime
- properties:
- monitors:
- description: 'monitors is Required: Monitors is a collection
- of Ceph monitors More info: https://examples.k8s.io/volumes/cephfs/README.md#how-to-use-it'
- items:
- type: string
- type: array
- path:
- description: 'path is Optional: Used as the mounted root,
- rather than the full Ceph tree, default is /'
- type: string
- readOnly:
- description: 'readOnly is Optional: Defaults to false (read/write).
- ReadOnly here will force the ReadOnly setting in VolumeMounts.
- More info: https://examples.k8s.io/volumes/cephfs/README.md#how-to-use-it'
- type: boolean
- secretFile:
- description: 'secretFile is Optional: SecretFile is the
- path to key ring for User, default is /etc/ceph/user.secret
- More info: https://examples.k8s.io/volumes/cephfs/README.md#how-to-use-it'
- type: string
- secretRef:
- description: 'secretRef is Optional: SecretRef is reference
- to the authentication secret for User, default is empty.
- More info: https://examples.k8s.io/volumes/cephfs/README.md#how-to-use-it'
- properties:
- name:
- description: 'Name of the referent. More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names
- TODO: Add other useful fields. apiVersion, kind, uid?'
- type: string
- type: object
- user:
- description: 'user is optional: User is the rados user name,
- default is admin More info: https://examples.k8s.io/volumes/cephfs/README.md#how-to-use-it'
- type: string
- required:
- - monitors
- type: object
- cinder:
- description: 'cinder represents a cinder volume attached and
- mounted on kubelets host machine. More info: https://examples.k8s.io/mysql-cinder-pd/README.md'
- properties:
- fsType:
- description: 'fsType is the filesystem type to mount. Must
- be a filesystem type supported by the host operating system.
- Examples: "ext4", "xfs", "ntfs". Implicitly inferred to
- be "ext4" if unspecified. More info: https://examples.k8s.io/mysql-cinder-pd/README.md'
- type: string
- readOnly:
- description: 'readOnly defaults to false (read/write). ReadOnly
- here will force the ReadOnly setting in VolumeMounts.
- More info: https://examples.k8s.io/mysql-cinder-pd/README.md'
- type: boolean
- secretRef:
- description: 'secretRef is optional: points to a secret
- object containing parameters used to connect to OpenStack.'
- properties:
- name:
- description: 'Name of the referent. More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names
- TODO: Add other useful fields. apiVersion, kind, uid?'
- type: string
- type: object
- volumeID:
- description: 'volumeID used to identify the volume in cinder.
- More info: https://examples.k8s.io/mysql-cinder-pd/README.md'
- type: string
- required:
- - volumeID
- type: object
- configMap:
- description: configMap represents a configMap that should populate
- this volume
- properties:
- defaultMode:
- description: 'defaultMode is optional: mode bits used to
- set permissions on created files by default. Must be an
- octal value between 0000 and 0777 or a decimal value between
- 0 and 511. YAML accepts both octal and decimal values,
- JSON requires decimal values for mode bits. Defaults to
- 0644. Directories within the path are not affected by
- this setting. This might be in conflict with other options
- that affect the file mode, like fsGroup, and the result
- can be other mode bits set.'
- format: int32
- type: integer
- items:
- description: items if unspecified, each key-value pair in
- the Data field of the referenced ConfigMap will be projected
- into the volume as a file whose name is the key and content
- is the value. If specified, the listed keys will be projected
- into the specified paths, and unlisted keys will not be
- present. If a key is specified which is not present in
- the ConfigMap, the volume setup will error unless it is
- marked optional. Paths must be relative and may not contain
- the '..' path or start with '..'.
- items:
- description: Maps a string key to a path within a volume.
- properties:
- key:
- description: key is the key to project.
- type: string
- mode:
- description: 'mode is Optional: mode bits used to
- set permissions on this file. Must be an octal value
- between 0000 and 0777 or a decimal value between
- 0 and 511. YAML accepts both octal and decimal values,
- JSON requires decimal values for mode bits. If not
- specified, the volume defaultMode will be used.
- This might be in conflict with other options that
- affect the file mode, like fsGroup, and the result
- can be other mode bits set.'
- format: int32
- type: integer
- path:
- description: path is the relative path of the file
- to map the key to. May not be an absolute path.
- May not contain the path element '..'. May not start
- with the string '..'.
- type: string
- required:
- - key
- - path
- type: object
- type: array
- name:
- description: 'Name of the referent. More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names
- TODO: Add other useful fields. apiVersion, kind, uid?'
- type: string
- optional:
- description: optional specify whether the ConfigMap or its
- keys must be defined
- type: boolean
- type: object
- csi:
- description: csi (Container Storage Interface) represents ephemeral
- storage that is handled by certain external CSI drivers (Beta
- feature).
- properties:
- driver:
- description: driver is the name of the CSI driver that handles
- this volume. Consult with your admin for the correct name
- as registered in the cluster.
- type: string
- fsType:
- description: fsType to mount. Ex. "ext4", "xfs", "ntfs".
- If not provided, the empty value is passed to the associated
- CSI driver which will determine the default filesystem
- to apply.
- type: string
- nodePublishSecretRef:
- description: nodePublishSecretRef is a reference to the
- secret object containing sensitive information to pass
- to the CSI driver to complete the CSI NodePublishVolume
- and NodeUnpublishVolume calls. This field is optional,
- and may be empty if no secret is required. If the secret
- object contains more than one secret, all secret references
- are passed.
- properties:
- name:
- description: 'Name of the referent. More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names
- TODO: Add other useful fields. apiVersion, kind, uid?'
- type: string
- type: object
- readOnly:
- description: readOnly specifies a read-only configuration
- for the volume. Defaults to false (read/write).
- type: boolean
- volumeAttributes:
- additionalProperties:
- type: string
- description: volumeAttributes stores driver-specific properties
- that are passed to the CSI driver. Consult your driver's
- documentation for supported values.
- type: object
- required:
- - driver
- type: object
- downwardAPI:
- description: downwardAPI represents downward API about the pod
- that should populate this volume
- properties:
- defaultMode:
- description: 'Optional: mode bits to use on created files
- by default. Must be a Optional: mode bits used to set
- permissions on created files by default. Must be an octal
- value between 0000 and 0777 or a decimal value between
- 0 and 511. YAML accepts both octal and decimal values,
- JSON requires decimal values for mode bits. Defaults to
- 0644. Directories within the path are not affected by
- this setting. This might be in conflict with other options
- that affect the file mode, like fsGroup, and the result
- can be other mode bits set.'
- format: int32
- type: integer
- items:
- description: Items is a list of downward API volume file
- items:
- description: DownwardAPIVolumeFile represents information
- to create the file containing the pod field
- properties:
- fieldRef:
- description: 'Required: Selects a field of the pod:
- only annotations, labels, name and namespace are
- supported.'
- properties:
- apiVersion:
- description: Version of the schema the FieldPath
- is written in terms of, defaults to "v1".
- type: string
- fieldPath:
- description: Path of the field to select in the
- specified API version.
- type: string
- required:
- - fieldPath
- type: object
- mode:
- description: 'Optional: mode bits used to set permissions
- on this file, must be an octal value between 0000
- and 0777 or a decimal value between 0 and 511. YAML
- accepts both octal and decimal values, JSON requires
- decimal values for mode bits. If not specified,
- the volume defaultMode will be used. This might
- be in conflict with other options that affect the
- file mode, like fsGroup, and the result can be other
- mode bits set.'
- format: int32
- type: integer
- path:
- description: 'Required: Path is the relative path
- name of the file to be created. Must not be absolute
- or contain the ''..'' path. Must be utf-8 encoded.
- The first item of the relative path must not start
- with ''..'''
- type: string
- resourceFieldRef:
- description: 'Selects a resource of the container:
- only resources limits and requests (limits.cpu,
- limits.memory, requests.cpu and requests.memory)
- are currently supported.'
- properties:
- containerName:
- description: 'Container name: required for volumes,
- optional for env vars'
- type: string
- divisor:
- anyOf:
- - type: integer
- - type: string
- description: Specifies the output format of the
- exposed resources, defaults to "1"
- pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$
- x-kubernetes-int-or-string: true
- resource:
- description: 'Required: resource to select'
- type: string
- required:
- - resource
- type: object
- required:
- - path
- type: object
- type: array
- type: object
- emptyDir:
- description: 'emptyDir represents a temporary directory that
- shares a pod''s lifetime. More info: https://kubernetes.io/docs/concepts/storage/volumes#emptydir'
- properties:
- medium:
- description: 'medium represents what type of storage medium
- should back this directory. The default is "" which means
- to use the node''s default medium. Must be an empty string
- (default) or Memory. More info: https://kubernetes.io/docs/concepts/storage/volumes#emptydir'
- type: string
- sizeLimit:
- anyOf:
- - type: integer
- - type: string
- description: 'sizeLimit is the total amount of local storage
- required for this EmptyDir volume. The size limit is also
- applicable for memory medium. The maximum usage on memory
- medium EmptyDir would be the minimum value between the
- SizeLimit specified here and the sum of memory limits
- of all containers in a pod. The default is nil which means
- that the limit is undefined. More info: http://kubernetes.io/docs/user-guide/volumes#emptydir'
- pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$
- x-kubernetes-int-or-string: true
- type: object
- ephemeral:
- description: "ephemeral represents a volume that is handled
- by a cluster storage driver. The volume's lifecycle is tied
- to the pod that defines it - it will be created before the
- pod starts, and deleted when the pod is removed. \n Use this
- if: a) the volume is only needed while the pod runs, b) features
- of normal volumes like restoring from snapshot or capacity
- tracking are needed, c) the storage driver is specified through
- a storage class, and d) the storage driver supports dynamic
- volume provisioning through a PersistentVolumeClaim (see EphemeralVolumeSource
- for more information on the connection between this volume
- type and PersistentVolumeClaim). \n Use PersistentVolumeClaim
- or one of the vendor-specific APIs for volumes that persist
- for longer than the lifecycle of an individual pod. \n Use
- CSI for light-weight local ephemeral volumes if the CSI driver
- is meant to be used that way - see the documentation of the
- driver for more information. \n A pod can use both types of
- ephemeral volumes and persistent volumes at the same time."
- properties:
- volumeClaimTemplate:
- description: "Will be used to create a stand-alone PVC to
- provision the volume. The pod in which this EphemeralVolumeSource
- is embedded will be the owner of the PVC, i.e. the PVC
- will be deleted together with the pod. The name of the
- PVC will be `-` where `` is the name from the `PodSpec.Volumes` array entry.
- Pod validation will reject the pod if the concatenated
- name is not valid for a PVC (for example, too long). \n
- An existing PVC with that name that is not owned by the
- pod will *not* be used for the pod to avoid using an unrelated
- volume by mistake. Starting the pod is then blocked until
- the unrelated PVC is removed. If such a pre-created PVC
- is meant to be used by the pod, the PVC has to updated
- with an owner reference to the pod once the pod exists.
- Normally this should not be necessary, but it may be useful
- when manually reconstructing a broken cluster. \n This
- field is read-only and no changes will be made by Kubernetes
- to the PVC after it has been created. \n Required, must
- not be nil."
- properties:
- metadata:
- description: May contain labels and annotations that
- will be copied into the PVC when creating it. No other
- fields are allowed and will be rejected during validation.
- properties:
- annotations:
- additionalProperties:
- type: string
- type: object
- finalizers:
- items:
- type: string
- type: array
- labels:
- additionalProperties:
- type: string
- type: object
- name:
- type: string
- namespace:
- type: string
- type: object
- spec:
- description: The specification for the PersistentVolumeClaim.
- The entire content is copied unchanged into the PVC
- that gets created from this template. The same fields
- as in a PersistentVolumeClaim are also valid here.
- properties:
- accessModes:
- description: 'accessModes contains the desired access
- modes the volume should have. More info: https://kubernetes.io/docs/concepts/storage/persistent-volumes#access-modes-1'
- items:
- type: string
- type: array
- dataSource:
- description: 'dataSource field can be used to specify
- either: * An existing VolumeSnapshot object (snapshot.storage.k8s.io/VolumeSnapshot)
- * An existing PVC (PersistentVolumeClaim) If the
- provisioner or an external controller can support
- the specified data source, it will create a new
- volume based on the contents of the specified
- data source. When the AnyVolumeDataSource feature
- gate is enabled, dataSource contents will be copied
- to dataSourceRef, and dataSourceRef contents will
- be copied to dataSource when dataSourceRef.namespace
- is not specified. If the namespace is specified,
- then dataSourceRef will not be copied to dataSource.'
- properties:
- apiGroup:
- description: APIGroup is the group for the resource
- being referenced. If APIGroup is not specified,
- the specified Kind must be in the core API
- group. For any other third-party types, APIGroup
- is required.
- type: string
- kind:
- description: Kind is the type of resource being
- referenced
- type: string
- name:
- description: Name is the name of resource being
- referenced
- type: string
- required:
- - kind
- - name
- type: object
- dataSourceRef:
- description: 'dataSourceRef specifies the object
- from which to populate the volume with data, if
- a non-empty volume is desired. This may be any
- object from a non-empty API group (non core object)
- or a PersistentVolumeClaim object. When this field
- is specified, volume binding will only succeed
- if the type of the specified object matches some
- installed volume populator or dynamic provisioner.
- This field will replace the functionality of the
- dataSource field and as such if both fields are
- non-empty, they must have the same value. For
- backwards compatibility, when namespace isn''t
- specified in dataSourceRef, both fields (dataSource
- and dataSourceRef) will be set to the same value
- automatically if one of them is empty and the
- other is non-empty. When namespace is specified
- in dataSourceRef, dataSource isn''t set to the
- same value and must be empty. There are three
- important differences between dataSource and dataSourceRef:
- * While dataSource only allows two specific types
- of objects, dataSourceRef allows any non-core
- object, as well as PersistentVolumeClaim objects.
- * While dataSource ignores disallowed values (dropping
- them), dataSourceRef preserves all values, and
- generates an error if a disallowed value is specified.
- * While dataSource only allows local objects,
- dataSourceRef allows objects in any namespaces.
- (Beta) Using this field requires the AnyVolumeDataSource
- feature gate to be enabled. (Alpha) Using the
- namespace field of dataSourceRef requires the
- CrossNamespaceVolumeDataSource feature gate to
- be enabled.'
- properties:
- apiGroup:
- description: APIGroup is the group for the resource
- being referenced. If APIGroup is not specified,
- the specified Kind must be in the core API
- group. For any other third-party types, APIGroup
- is required.
- type: string
- kind:
- description: Kind is the type of resource being
- referenced
- type: string
- name:
- description: Name is the name of resource being
- referenced
- type: string
- namespace:
- description: Namespace is the namespace of resource
- being referenced Note that when a namespace
- is specified, a gateway.networking.k8s.io/ReferenceGrant
- object is required in the referent namespace
- to allow that namespace's owner to accept
- the reference. See the ReferenceGrant documentation
- for details. (Alpha) This field requires the
- CrossNamespaceVolumeDataSource feature gate
- to be enabled.
- type: string
- required:
- - kind
- - name
- type: object
- resources:
- description: 'resources represents the minimum resources
- the volume should have. If RecoverVolumeExpansionFailure
- feature is enabled users are allowed to specify
- resource requirements that are lower than previous
- value but must still be higher than capacity recorded
- in the status field of the claim. More info: https://kubernetes.io/docs/concepts/storage/persistent-volumes#resources'
- properties:
- claims:
- description: "Claims lists the names of resources,
- defined in spec.resourceClaims, that are used
- by this container. \n This is an alpha field
- and requires enabling the DynamicResourceAllocation
- feature gate. \n This field is immutable.
- It can only be set for containers."
- items:
- description: ResourceClaim references one
- entry in PodSpec.ResourceClaims.
- properties:
- name:
- description: Name must match the name
- of one entry in pod.spec.resourceClaims
- of the Pod where this field is used.
- It makes that resource available inside
- a container.
- type: string
- required:
- - name
- type: object
- type: array
- x-kubernetes-list-map-keys:
- - name
- x-kubernetes-list-type: map
- limits:
- additionalProperties:
- anyOf:
- - type: integer
- - type: string
- pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$
- x-kubernetes-int-or-string: true
- description: 'Limits describes the maximum amount
- of compute resources allowed. More info: https://kubernetes.io/docs/concepts/configuration/manage-resources-containers/'
- type: object
- requests:
- additionalProperties:
- anyOf:
- - type: integer
- - type: string
- pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$
- x-kubernetes-int-or-string: true
- description: 'Requests describes the minimum
- amount of compute resources required. If Requests
- is omitted for a container, it defaults to
- Limits if that is explicitly specified, otherwise
- to an implementation-defined value. More info:
- https://kubernetes.io/docs/concepts/configuration/manage-resources-containers/'
- type: object
- type: object
- selector:
- description: selector is a label query over volumes
- to consider for binding.
- properties:
- matchExpressions:
- description: matchExpressions is a list of label
- selector requirements. The requirements are
- ANDed.
- items:
- description: A label selector requirement
- is a selector that contains values, a key,
- and an operator that relates the key and
- values.
- properties:
- key:
- description: key is the label key that
- the selector applies to.
- type: string
- operator:
- description: operator represents a key's
- relationship to a set of values. Valid
- operators are In, NotIn, Exists and
- DoesNotExist.
- type: string
- values:
- description: values is an array of string
- values. If the operator is In or NotIn,
- the values array must be non-empty.
- If the operator is Exists or DoesNotExist,
- the values array must be empty. This
- array is replaced during a strategic
- merge patch.
- items:
- type: string
- type: array
- required:
- - key
- - operator
- type: object
- type: array
- matchLabels:
- additionalProperties:
- type: string
- description: matchLabels is a map of {key,value}
- pairs. A single {key,value} in the matchLabels
- map is equivalent to an element of matchExpressions,
- whose key field is "key", the operator is
- "In", and the values array contains only "value".
- The requirements are ANDed.
- type: object
- type: object
- storageClassName:
- description: 'storageClassName is the name of the
- StorageClass required by the claim. More info:
- https://kubernetes.io/docs/concepts/storage/persistent-volumes#class-1'
- type: string
- volumeMode:
- description: volumeMode defines what type of volume
- is required by the claim. Value of Filesystem
- is implied when not included in claim spec.
- type: string
- volumeName:
- description: volumeName is the binding reference
- to the PersistentVolume backing this claim.
- type: string
- type: object
- required:
- - spec
- type: object
- type: object
- fc:
- description: fc represents a Fibre Channel resource that is
- attached to a kubelet's host machine and then exposed to the
- pod.
- properties:
- fsType:
- description: 'fsType is the filesystem type to mount. Must
- be a filesystem type supported by the host operating system.
- Ex. "ext4", "xfs", "ntfs". Implicitly inferred to be "ext4"
- if unspecified. TODO: how do we prevent errors in the
- filesystem from compromising the machine'
- type: string
- lun:
- description: 'lun is Optional: FC target lun number'
- format: int32
- type: integer
- readOnly:
- description: 'readOnly is Optional: Defaults to false (read/write).
- ReadOnly here will force the ReadOnly setting in VolumeMounts.'
- type: boolean
- targetWWNs:
- description: 'targetWWNs is Optional: FC target worldwide
- names (WWNs)'
- items:
- type: string
- type: array
- wwids:
- description: 'wwids Optional: FC volume world wide identifiers
- (wwids) Either wwids or combination of targetWWNs and
- lun must be set, but not both simultaneously.'
- items:
- type: string
- type: array
- type: object
- flexVolume:
- description: flexVolume represents a generic volume resource
- that is provisioned/attached using an exec based plugin.
- properties:
- driver:
- description: driver is the name of the driver to use for
- this volume.
- type: string
- fsType:
- description: fsType is the filesystem type to mount. Must
- be a filesystem type supported by the host operating system.
- Ex. "ext4", "xfs", "ntfs". The default filesystem depends
- on FlexVolume script.
- type: string
- options:
- additionalProperties:
- type: string
- description: 'options is Optional: this field holds extra
- command options if any.'
- type: object
- readOnly:
- description: 'readOnly is Optional: defaults to false (read/write).
- ReadOnly here will force the ReadOnly setting in VolumeMounts.'
- type: boolean
- secretRef:
- description: 'secretRef is Optional: secretRef is reference
- to the secret object containing sensitive information
- to pass to the plugin scripts. This may be empty if no
- secret object is specified. If the secret object contains
- more than one secret, all secrets are passed to the plugin
- scripts.'
- properties:
- name:
- description: 'Name of the referent. More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names
- TODO: Add other useful fields. apiVersion, kind, uid?'
- type: string
- type: object
- required:
- - driver
- type: object
- flocker:
- description: flocker represents a Flocker volume attached to
- a kubelet's host machine. This depends on the Flocker control
- service being running
- properties:
- datasetName:
- description: datasetName is Name of the dataset stored as
- metadata -> name on the dataset for Flocker should be
- considered as deprecated
- type: string
- datasetUUID:
- description: datasetUUID is the UUID of the dataset. This
- is unique identifier of a Flocker dataset
- type: string
- type: object
- gcePersistentDisk:
- description: 'gcePersistentDisk represents a GCE Disk resource
- that is attached to a kubelet''s host machine and then exposed
- to the pod. More info: https://kubernetes.io/docs/concepts/storage/volumes#gcepersistentdisk'
- properties:
- fsType:
- description: 'fsType is filesystem type of the volume that
- you want to mount. Tip: Ensure that the filesystem type
- is supported by the host operating system. Examples: "ext4",
- "xfs", "ntfs". Implicitly inferred to be "ext4" if unspecified.
- More info: https://kubernetes.io/docs/concepts/storage/volumes#gcepersistentdisk
- TODO: how do we prevent errors in the filesystem from
- compromising the machine'
- type: string
- partition:
- description: 'partition is the partition in the volume that
- you want to mount. If omitted, the default is to mount
- by volume name. Examples: For volume /dev/sda1, you specify
- the partition as "1". Similarly, the volume partition
- for /dev/sda is "0" (or you can leave the property empty).
- More info: https://kubernetes.io/docs/concepts/storage/volumes#gcepersistentdisk'
- format: int32
- type: integer
- pdName:
- description: 'pdName is unique name of the PD resource in
- GCE. Used to identify the disk in GCE. More info: https://kubernetes.io/docs/concepts/storage/volumes#gcepersistentdisk'
- type: string
- readOnly:
- description: 'readOnly here will force the ReadOnly setting
- in VolumeMounts. Defaults to false. More info: https://kubernetes.io/docs/concepts/storage/volumes#gcepersistentdisk'
- type: boolean
- required:
- - pdName
- type: object
- gitRepo:
- description: 'gitRepo represents a git repository at a particular
- revision. DEPRECATED: GitRepo is deprecated. To provision
- a container with a git repo, mount an EmptyDir into an InitContainer
- that clones the repo using git, then mount the EmptyDir into
- the Pod''s container.'
- properties:
- directory:
- description: directory is the target directory name. Must
- not contain or start with '..'. If '.' is supplied, the
- volume directory will be the git repository. Otherwise,
- if specified, the volume will contain the git repository
- in the subdirectory with the given name.
- type: string
- repository:
- description: repository is the URL
- type: string
- revision:
- description: revision is the commit hash for the specified
- revision.
- type: string
- required:
- - repository
- type: object
- glusterfs:
- description: 'glusterfs represents a Glusterfs mount on the
- host that shares a pod''s lifetime. More info: https://examples.k8s.io/volumes/glusterfs/README.md'
- properties:
- endpoints:
- description: 'endpoints is the endpoint name that details
- Glusterfs topology. More info: https://examples.k8s.io/volumes/glusterfs/README.md#create-a-pod'
- type: string
- path:
- description: 'path is the Glusterfs volume path. More info:
- https://examples.k8s.io/volumes/glusterfs/README.md#create-a-pod'
- type: string
- readOnly:
- description: 'readOnly here will force the Glusterfs volume
- to be mounted with read-only permissions. Defaults to
- false. More info: https://examples.k8s.io/volumes/glusterfs/README.md#create-a-pod'
- type: boolean
- required:
- - endpoints
- - path
- type: object
- hostPath:
- description: 'hostPath represents a pre-existing file or directory
- on the host machine that is directly exposed to the container.
- This is generally used for system agents or other privileged
- things that are allowed to see the host machine. Most containers
- will NOT need this. More info: https://kubernetes.io/docs/concepts/storage/volumes#hostpath
- --- TODO(jonesdl) We need to restrict who can use host directory
- mounts and who can/can not mount host directories as read/write.'
- properties:
- path:
- description: 'path of the directory on the host. If the
- path is a symlink, it will follow the link to the real
- path. More info: https://kubernetes.io/docs/concepts/storage/volumes#hostpath'
- type: string
- type:
- description: 'type for HostPath Volume Defaults to "" More
- info: https://kubernetes.io/docs/concepts/storage/volumes#hostpath'
- type: string
- required:
- - path
- type: object
- iscsi:
- description: 'iscsi represents an ISCSI Disk resource that is
- attached to a kubelet''s host machine and then exposed to
- the pod. More info: https://examples.k8s.io/volumes/iscsi/README.md'
- properties:
- chapAuthDiscovery:
- description: chapAuthDiscovery defines whether support iSCSI
- Discovery CHAP authentication
- type: boolean
- chapAuthSession:
- description: chapAuthSession defines whether support iSCSI
- Session CHAP authentication
- type: boolean
- fsType:
- description: 'fsType is the filesystem type of the volume
- that you want to mount. Tip: Ensure that the filesystem
- type is supported by the host operating system. Examples:
- "ext4", "xfs", "ntfs". Implicitly inferred to be "ext4"
- if unspecified. More info: https://kubernetes.io/docs/concepts/storage/volumes#iscsi
- TODO: how do we prevent errors in the filesystem from
- compromising the machine'
- type: string
- initiatorName:
- description: initiatorName is the custom iSCSI Initiator
- Name. If initiatorName is specified with iscsiInterface
- simultaneously, new iSCSI interface : will be created for the connection.
- type: string
- iqn:
- description: iqn is the target iSCSI Qualified Name.
- type: string
- iscsiInterface:
- description: iscsiInterface is the interface Name that uses
- an iSCSI transport. Defaults to 'default' (tcp).
- type: string
- lun:
- description: lun represents iSCSI Target Lun number.
- format: int32
- type: integer
- portals:
- description: portals is the iSCSI Target Portal List. The
- portal is either an IP or ip_addr:port if the port is
- other than default (typically TCP ports 860 and 3260).
- items:
- type: string
- type: array
- readOnly:
- description: readOnly here will force the ReadOnly setting
- in VolumeMounts. Defaults to false.
- type: boolean
- secretRef:
- description: secretRef is the CHAP Secret for iSCSI target
- and initiator authentication
- properties:
- name:
- description: 'Name of the referent. More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names
- TODO: Add other useful fields. apiVersion, kind, uid?'
- type: string
- type: object
- targetPortal:
- description: targetPortal is iSCSI Target Portal. The Portal
- is either an IP or ip_addr:port if the port is other than
- default (typically TCP ports 860 and 3260).
- type: string
- required:
- - iqn
- - lun
- - targetPortal
- type: object
- name:
- description: 'name of the volume. Must be a DNS_LABEL and unique
- within the pod. More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names'
- type: string
- nfs:
- description: 'nfs represents an NFS mount on the host that shares
- a pod''s lifetime More info: https://kubernetes.io/docs/concepts/storage/volumes#nfs'
- properties:
- path:
- description: 'path that is exported by the NFS server. More
- info: https://kubernetes.io/docs/concepts/storage/volumes#nfs'
- type: string
- readOnly:
- description: 'readOnly here will force the NFS export to
- be mounted with read-only permissions. Defaults to false.
- More info: https://kubernetes.io/docs/concepts/storage/volumes#nfs'
- type: boolean
- server:
- description: 'server is the hostname or IP address of the
- NFS server. More info: https://kubernetes.io/docs/concepts/storage/volumes#nfs'
- type: string
- required:
- - path
- - server
- type: object
- persistentVolumeClaim:
- description: 'persistentVolumeClaimVolumeSource represents a
- reference to a PersistentVolumeClaim in the same namespace.
- More info: https://kubernetes.io/docs/concepts/storage/persistent-volumes#persistentvolumeclaims'
- properties:
- claimName:
- description: 'claimName is the name of a PersistentVolumeClaim
- in the same namespace as the pod using this volume. More
- info: https://kubernetes.io/docs/concepts/storage/persistent-volumes#persistentvolumeclaims'
- type: string
- readOnly:
- description: readOnly Will force the ReadOnly setting in
- VolumeMounts. Default false.
- type: boolean
- required:
- - claimName
- type: object
- photonPersistentDisk:
- description: photonPersistentDisk represents a PhotonController
- persistent disk attached and mounted on kubelets host machine
- properties:
- fsType:
- description: fsType is the filesystem type to mount. Must
- be a filesystem type supported by the host operating system.
- Ex. "ext4", "xfs", "ntfs". Implicitly inferred to be "ext4"
- if unspecified.
- type: string
- pdID:
- description: pdID is the ID that identifies Photon Controller
- persistent disk
- type: string
- required:
- - pdID
- type: object
- portworxVolume:
- description: portworxVolume represents a portworx volume attached
- and mounted on kubelets host machine
- properties:
- fsType:
- description: fSType represents the filesystem type to mount
- Must be a filesystem type supported by the host operating
- system. Ex. "ext4", "xfs". Implicitly inferred to be "ext4"
- if unspecified.
- type: string
- readOnly:
- description: readOnly defaults to false (read/write). ReadOnly
- here will force the ReadOnly setting in VolumeMounts.
- type: boolean
- volumeID:
- description: volumeID uniquely identifies a Portworx volume
- type: string
- required:
- - volumeID
- type: object
- projected:
- description: projected items for all in one resources secrets,
- configmaps, and downward API
- properties:
- defaultMode:
- description: defaultMode are the mode bits used to set permissions
- on created files by default. Must be an octal value between
- 0000 and 0777 or a decimal value between 0 and 511. YAML
- accepts both octal and decimal values, JSON requires decimal
- values for mode bits. Directories within the path are
- not affected by this setting. This might be in conflict
- with other options that affect the file mode, like fsGroup,
- and the result can be other mode bits set.
- format: int32
- type: integer
- sources:
- description: sources is the list of volume projections
- items:
- description: Projection that may be projected along with
- other supported volume types
- properties:
- configMap:
- description: configMap information about the configMap
- data to project
- properties:
- items:
- description: items if unspecified, each key-value
- pair in the Data field of the referenced ConfigMap
- will be projected into the volume as a file
- whose name is the key and content is the value.
- If specified, the listed keys will be projected
- into the specified paths, and unlisted keys
- will not be present. If a key is specified which
- is not present in the ConfigMap, the volume
- setup will error unless it is marked optional.
- Paths must be relative and may not contain the
- '..' path or start with '..'.
- items:
- description: Maps a string key to a path within
- a volume.
- properties:
- key:
- description: key is the key to project.
- type: string
- mode:
- description: 'mode is Optional: mode bits
- used to set permissions on this file.
- Must be an octal value between 0000 and
- 0777 or a decimal value between 0 and
- 511. YAML accepts both octal and decimal
- values, JSON requires decimal values for
- mode bits. If not specified, the volume
- defaultMode will be used. This might be
- in conflict with other options that affect
- the file mode, like fsGroup, and the result
- can be other mode bits set.'
- format: int32
- type: integer
- path:
- description: path is the relative path of
- the file to map the key to. May not be
- an absolute path. May not contain the
- path element '..'. May not start with
- the string '..'.
- type: string
- required:
- - key
- - path
- type: object
- type: array
- name:
- description: 'Name of the referent. More info:
- https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names
- TODO: Add other useful fields. apiVersion, kind,
- uid?'
- type: string
- optional:
- description: optional specify whether the ConfigMap
- or its keys must be defined
- type: boolean
- type: object
- downwardAPI:
- description: downwardAPI information about the downwardAPI
- data to project
- properties:
- items:
- description: Items is a list of DownwardAPIVolume
- file
- items:
- description: DownwardAPIVolumeFile represents
- information to create the file containing
- the pod field
- properties:
- fieldRef:
- description: 'Required: Selects a field
- of the pod: only annotations, labels,
- name and namespace are supported.'
- properties:
- apiVersion:
- description: Version of the schema the
- FieldPath is written in terms of,
- defaults to "v1".
- type: string
- fieldPath:
- description: Path of the field to select
- in the specified API version.
- type: string
- required:
- - fieldPath
- type: object
- mode:
- description: 'Optional: mode bits used to
- set permissions on this file, must be
- an octal value between 0000 and 0777 or
- a decimal value between 0 and 511. YAML
- accepts both octal and decimal values,
- JSON requires decimal values for mode
- bits. If not specified, the volume defaultMode
- will be used. This might be in conflict
- with other options that affect the file
- mode, like fsGroup, and the result can
- be other mode bits set.'
- format: int32
- type: integer
- path:
- description: 'Required: Path is the relative
- path name of the file to be created. Must
- not be absolute or contain the ''..''
- path. Must be utf-8 encoded. The first
- item of the relative path must not start
- with ''..'''
- type: string
- resourceFieldRef:
- description: 'Selects a resource of the
- container: only resources limits and requests
- (limits.cpu, limits.memory, requests.cpu
- and requests.memory) are currently supported.'
- properties:
- containerName:
- description: 'Container name: required
- for volumes, optional for env vars'
- type: string
- divisor:
- anyOf:
- - type: integer
- - type: string
- description: Specifies the output format
- of the exposed resources, defaults
- to "1"
- pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$
- x-kubernetes-int-or-string: true
- resource:
- description: 'Required: resource to
- select'
- type: string
- required:
- - resource
- type: object
- required:
- - path
- type: object
- type: array
- type: object
- secret:
- description: secret information about the secret data
- to project
- properties:
- items:
- description: items if unspecified, each key-value
- pair in the Data field of the referenced Secret
- will be projected into the volume as a file
- whose name is the key and content is the value.
- If specified, the listed keys will be projected
- into the specified paths, and unlisted keys
- will not be present. If a key is specified which
- is not present in the Secret, the volume setup
- will error unless it is marked optional. Paths
- must be relative and may not contain the '..'
- path or start with '..'.
- items:
- description: Maps a string key to a path within
- a volume.
- properties:
- key:
- description: key is the key to project.
- type: string
- mode:
- description: 'mode is Optional: mode bits
- used to set permissions on this file.
- Must be an octal value between 0000 and
- 0777 or a decimal value between 0 and
- 511. YAML accepts both octal and decimal
- values, JSON requires decimal values for
- mode bits. If not specified, the volume
- defaultMode will be used. This might be
- in conflict with other options that affect
- the file mode, like fsGroup, and the result
- can be other mode bits set.'
- format: int32
- type: integer
- path:
- description: path is the relative path of
- the file to map the key to. May not be
- an absolute path. May not contain the
- path element '..'. May not start with
- the string '..'.
- type: string
- required:
- - key
- - path
- type: object
- type: array
- name:
- description: 'Name of the referent. More info:
- https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names
- TODO: Add other useful fields. apiVersion, kind,
- uid?'
- type: string
- optional:
- description: optional field specify whether the
- Secret or its key must be defined
- type: boolean
- type: object
- serviceAccountToken:
- description: serviceAccountToken is information about
- the serviceAccountToken data to project
- properties:
- audience:
- description: audience is the intended audience
- of the token. A recipient of a token must identify
- itself with an identifier specified in the audience
- of the token, and otherwise should reject the
- token. The audience defaults to the identifier
- of the apiserver.
- type: string
- expirationSeconds:
- description: expirationSeconds is the requested
- duration of validity of the service account
- token. As the token approaches expiration, the
- kubelet volume plugin will proactively rotate
- the service account token. The kubelet will
- start trying to rotate the token if the token
- is older than 80 percent of its time to live
- or if the token is older than 24 hours.Defaults
- to 1 hour and must be at least 10 minutes.
- format: int64
- type: integer
- path:
- description: path is the path relative to the
- mount point of the file to project the token
- into.
- type: string
- required:
- - path
- type: object
- type: object
- type: array
- type: object
- quobyte:
- description: quobyte represents a Quobyte mount on the host
- that shares a pod's lifetime
- properties:
- group:
- description: group to map volume access to Default is no
- group
- type: string
- readOnly:
- description: readOnly here will force the Quobyte volume
- to be mounted with read-only permissions. Defaults to
- false.
- type: boolean
- registry:
- description: registry represents a single or multiple Quobyte
- Registry services specified as a string as host:port pair
- (multiple entries are separated with commas) which acts
- as the central registry for volumes
- type: string
- tenant:
- description: tenant owning the given Quobyte volume in the
- Backend Used with dynamically provisioned Quobyte volumes,
- value is set by the plugin
- type: string
- user:
- description: user to map volume access to Defaults to serivceaccount
- user
- type: string
- volume:
- description: volume is a string that references an already
- created Quobyte volume by name.
- type: string
- required:
- - registry
- - volume
- type: object
- rbd:
- description: 'rbd represents a Rados Block Device mount on the
- host that shares a pod''s lifetime. More info: https://examples.k8s.io/volumes/rbd/README.md'
- properties:
- fsType:
- description: 'fsType is the filesystem type of the volume
- that you want to mount. Tip: Ensure that the filesystem
- type is supported by the host operating system. Examples:
- "ext4", "xfs", "ntfs". Implicitly inferred to be "ext4"
- if unspecified. More info: https://kubernetes.io/docs/concepts/storage/volumes#rbd
- TODO: how do we prevent errors in the filesystem from
- compromising the machine'
- type: string
- image:
- description: 'image is the rados image name. More info:
- https://examples.k8s.io/volumes/rbd/README.md#how-to-use-it'
- type: string
- keyring:
- description: 'keyring is the path to key ring for RBDUser.
- Default is /etc/ceph/keyring. More info: https://examples.k8s.io/volumes/rbd/README.md#how-to-use-it'
- type: string
- monitors:
- description: 'monitors is a collection of Ceph monitors.
- More info: https://examples.k8s.io/volumes/rbd/README.md#how-to-use-it'
- items:
- type: string
- type: array
- pool:
- description: 'pool is the rados pool name. Default is rbd.
- More info: https://examples.k8s.io/volumes/rbd/README.md#how-to-use-it'
- type: string
- readOnly:
- description: 'readOnly here will force the ReadOnly setting
- in VolumeMounts. Defaults to false. More info: https://examples.k8s.io/volumes/rbd/README.md#how-to-use-it'
- type: boolean
- secretRef:
- description: 'secretRef is name of the authentication secret
- for RBDUser. If provided overrides keyring. Default is
- nil. More info: https://examples.k8s.io/volumes/rbd/README.md#how-to-use-it'
- properties:
- name:
- description: 'Name of the referent. More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names
- TODO: Add other useful fields. apiVersion, kind, uid?'
- type: string
- type: object
- user:
- description: 'user is the rados user name. Default is admin.
- More info: https://examples.k8s.io/volumes/rbd/README.md#how-to-use-it'
- type: string
- required:
- - image
- - monitors
- type: object
- scaleIO:
- description: scaleIO represents a ScaleIO persistent volume
- attached and mounted on Kubernetes nodes.
- properties:
- fsType:
- description: fsType is the filesystem type to mount. Must
- be a filesystem type supported by the host operating system.
- Ex. "ext4", "xfs", "ntfs". Default is "xfs".
- type: string
- gateway:
- description: gateway is the host address of the ScaleIO
- API Gateway.
- type: string
- protectionDomain:
- description: protectionDomain is the name of the ScaleIO
- Protection Domain for the configured storage.
- type: string
- readOnly:
- description: readOnly Defaults to false (read/write). ReadOnly
- here will force the ReadOnly setting in VolumeMounts.
- type: boolean
- secretRef:
- description: secretRef references to the secret for ScaleIO
- user and other sensitive information. If this is not provided,
- Login operation will fail.
- properties:
- name:
- description: 'Name of the referent. More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names
- TODO: Add other useful fields. apiVersion, kind, uid?'
- type: string
- type: object
- sslEnabled:
- description: sslEnabled Flag enable/disable SSL communication
- with Gateway, default false
- type: boolean
- storageMode:
- description: storageMode indicates whether the storage for
- a volume should be ThickProvisioned or ThinProvisioned.
- Default is ThinProvisioned.
- type: string
- storagePool:
- description: storagePool is the ScaleIO Storage Pool associated
- with the protection domain.
- type: string
- system:
- description: system is the name of the storage system as
- configured in ScaleIO.
- type: string
- volumeName:
- description: volumeName is the name of a volume already
- created in the ScaleIO system that is associated with
- this volume source.
- type: string
- required:
- - gateway
- - secretRef
- - system
- type: object
- secret:
- description: 'secret represents a secret that should populate
- this volume. More info: https://kubernetes.io/docs/concepts/storage/volumes#secret'
- properties:
- defaultMode:
- description: 'defaultMode is Optional: mode bits used to
- set permissions on created files by default. Must be an
- octal value between 0000 and 0777 or a decimal value between
- 0 and 511. YAML accepts both octal and decimal values,
- JSON requires decimal values for mode bits. Defaults to
- 0644. Directories within the path are not affected by
- this setting. This might be in conflict with other options
- that affect the file mode, like fsGroup, and the result
- can be other mode bits set.'
- format: int32
- type: integer
- items:
- description: items If unspecified, each key-value pair in
- the Data field of the referenced Secret will be projected
- into the volume as a file whose name is the key and content
- is the value. If specified, the listed keys will be projected
- into the specified paths, and unlisted keys will not be
- present. If a key is specified which is not present in
- the Secret, the volume setup will error unless it is marked
- optional. Paths must be relative and may not contain the
- '..' path or start with '..'.
- items:
- description: Maps a string key to a path within a volume.
- properties:
- key:
- description: key is the key to project.
- type: string
- mode:
- description: 'mode is Optional: mode bits used to
- set permissions on this file. Must be an octal value
- between 0000 and 0777 or a decimal value between
- 0 and 511. YAML accepts both octal and decimal values,
- JSON requires decimal values for mode bits. If not
- specified, the volume defaultMode will be used.
- This might be in conflict with other options that
- affect the file mode, like fsGroup, and the result
- can be other mode bits set.'
- format: int32
- type: integer
- path:
- description: path is the relative path of the file
- to map the key to. May not be an absolute path.
- May not contain the path element '..'. May not start
- with the string '..'.
- type: string
- required:
- - key
- - path
- type: object
- type: array
- optional:
- description: optional field specify whether the Secret or
- its keys must be defined
- type: boolean
- secretName:
- description: 'secretName is the name of the secret in the
- pod''s namespace to use. More info: https://kubernetes.io/docs/concepts/storage/volumes#secret'
- type: string
- type: object
- storageos:
- description: storageOS represents a StorageOS volume attached
- and mounted on Kubernetes nodes.
- properties:
- fsType:
- description: fsType is the filesystem type to mount. Must
- be a filesystem type supported by the host operating system.
- Ex. "ext4", "xfs", "ntfs". Implicitly inferred to be "ext4"
- if unspecified.
- type: string
- readOnly:
- description: readOnly defaults to false (read/write). ReadOnly
- here will force the ReadOnly setting in VolumeMounts.
- type: boolean
- secretRef:
- description: secretRef specifies the secret to use for obtaining
- the StorageOS API credentials. If not specified, default
- values will be attempted.
- properties:
- name:
- description: 'Name of the referent. More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names
- TODO: Add other useful fields. apiVersion, kind, uid?'
- type: string
- type: object
- volumeName:
- description: volumeName is the human-readable name of the
- StorageOS volume. Volume names are only unique within
- a namespace.
- type: string
- volumeNamespace:
- description: volumeNamespace specifies the scope of the
- volume within StorageOS. If no namespace is specified
- then the Pod's namespace will be used. This allows the
- Kubernetes name scoping to be mirrored within StorageOS
- for tighter integration. Set VolumeName to any name to
- override the default behaviour. Set to "default" if you
- are not using namespaces within StorageOS. Namespaces
- that do not pre-exist within StorageOS will be created.
- type: string
- type: object
- vsphereVolume:
- description: vsphereVolume represents a vSphere volume attached
- and mounted on kubelets host machine
- properties:
- fsType:
- description: fsType is filesystem type to mount. Must be
- a filesystem type supported by the host operating system.
- Ex. "ext4", "xfs", "ntfs". Implicitly inferred to be "ext4"
- if unspecified.
- type: string
- storagePolicyID:
- description: storagePolicyID is the storage Policy Based
- Management (SPBM) profile ID associated with the StoragePolicyName.
- type: string
- storagePolicyName:
- description: storagePolicyName is the storage Policy Based
- Management (SPBM) profile name.
- type: string
- volumePath:
- description: volumePath is the path that identifies vSphere
- volume vmdk
- type: string
- required:
- - volumePath
- type: object
- required:
- - name
- type: object
- minItems: 1
- type: array
- x-kubernetes-preserve-unknown-fields: true
- required:
- - backupJobName
- - target
- - targetVolumeMounts
- - targetVolumes
- type: object
- status:
- description: RestoreJobStatus defines the observed state of RestoreJob
- properties:
- completionTimestamp:
- description: Date/time when the backup finished being processed.
- format: date-time
- type: string
- expiration:
- description: The date and time when the Backup is eligible for garbage
- collection. 'null' means the Backup is NOT be cleaned except delete
- manual.
- format: date-time
- type: string
- failureReason:
- description: Job failed reason.
- type: string
- phase:
- description: RestoreJobPhase The current phase. Valid values are New,
- InProgressPhy, InProgressLogic, Completed, Failed.
- enum:
- - New
- - InProgressPhy
- - InProgressLogic
- - Completed
- - Failed
- type: string
- startTimestamp:
- description: Date/time when the backup started being processed.
- format: date-time
- type: string
- type: object
- type: object
- served: true
- storage: true
- subresources:
- status: {}
diff --git a/deploy/helm/crds/dataprotection.kubeblocks.io_restores.yaml b/deploy/helm/crds/dataprotection.kubeblocks.io_restores.yaml
new file mode 100644
index 00000000000..a36a9f042d4
--- /dev/null
+++ b/deploy/helm/crds/dataprotection.kubeblocks.io_restores.yaml
@@ -0,0 +1,2522 @@
+apiVersion: apiextensions.k8s.io/v1
+kind: CustomResourceDefinition
+metadata:
+ annotations:
+ controller-gen.kubebuilder.io/version: v0.12.1
+ labels:
+ app.kubernetes.io/name: kubeblocks
+ name: restores.dataprotection.kubeblocks.io
+spec:
+ group: dataprotection.kubeblocks.io
+ names:
+ categories:
+ - kubeblocks
+ - all
+ kind: Restore
+ listKind: RestoreList
+ plural: restores
+ singular: restore
+ scope: Namespaced
+ versions:
+ - additionalPrinterColumns:
+ - jsonPath: .spec.backup.name
+ name: BACKUP
+ type: string
+ - description: Point in time for restoring
+ jsonPath: .spec.restoreTime
+ name: RESTORE-TIME
+ type: string
+ - description: Restore Status.
+ jsonPath: .status.phase
+ name: STATUS
+ type: string
+ - jsonPath: .status.duration
+ name: DURATION
+ type: string
+ - jsonPath: .metadata.creationTimestamp
+ name: CREATE-TIME
+ type: string
+ - jsonPath: .status.completionTimestamp
+ name: COMPLETION-TIME
+ type: string
+ name: v1alpha1
+ schema:
+ openAPIV3Schema:
+ description: Restore is the Schema for the restores API
+ properties:
+ apiVersion:
+ description: 'APIVersion defines the versioned schema of this representation
+ of an object. Servers should convert recognized schemas to the latest
+ internal value, and may reject unrecognized values. More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#resources'
+ type: string
+ kind:
+ description: 'Kind is a string value representing the REST resource this
+ object represents. Servers may infer this from the endpoint the client
+ submits requests to. Cannot be updated. In CamelCase. More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#types-kinds'
+ type: string
+ metadata:
+ type: object
+ spec:
+ description: RestoreSpec defines the desired state of Restore
+ properties:
+ backup:
+ description: 'backup name, the following behavior based on the backup
+ type: 1. Full: will be restored the full backup directly. 2. Incremental:
+ will be restored sequentially from the most recent full backup of
+ this incremental backup. 3. Differential: will be restored sequentially
+ from the parent backup of the differential backup. 4. Continuous:
+ will find the most recent full backup at this time point and the
+ input continuous backup to restore.'
+ properties:
+ name:
+ description: backup name
+ type: string
+ namespace:
+ description: backup namespace
+ type: string
+ required:
+ - name
+ - namespace
+ type: object
+ x-kubernetes-validations:
+ - message: forbidden to update spec.backupName
+ rule: self == oldSelf
+ containerResources:
+ description: specified the required resources of restore job's container.
+ properties:
+ claims:
+ description: "Claims lists the names of resources, defined in
+ spec.resourceClaims, that are used by this container. \n This
+ is an alpha field and requires enabling the DynamicResourceAllocation
+ feature gate. \n This field is immutable. It can only be set
+ for containers."
+ items:
+ description: ResourceClaim references one entry in PodSpec.ResourceClaims.
+ properties:
+ name:
+ description: Name must match the name of one entry in pod.spec.resourceClaims
+ of the Pod where this field is used. It makes that resource
+ available inside a container.
+ type: string
+ required:
+ - name
+ type: object
+ type: array
+ x-kubernetes-list-map-keys:
+ - name
+ x-kubernetes-list-type: map
+ limits:
+ additionalProperties:
+ anyOf:
+ - type: integer
+ - type: string
+ pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$
+ x-kubernetes-int-or-string: true
+ description: 'Limits describes the maximum amount of compute resources
+ allowed. More info: https://kubernetes.io/docs/concepts/configuration/manage-resources-containers/'
+ type: object
+ requests:
+ additionalProperties:
+ anyOf:
+ - type: integer
+ - type: string
+ pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$
+ x-kubernetes-int-or-string: true
+ description: 'Requests describes the minimum amount of compute
+ resources required. If Requests is omitted for a container,
+ it defaults to Limits if that is explicitly specified, otherwise
+ to an implementation-defined value. Requests cannot exceed Limits.
+ More info: https://kubernetes.io/docs/concepts/configuration/manage-resources-containers/'
+ type: object
+ type: object
+ env:
+ description: 'list of environment variables to set in the container
+ for restore and will be merged with the env of Backup and ActionSet.
+ the priority of merging is as follows: Restore env > Backup env
+ > ActionSet env.'
+ items:
+ description: EnvVar represents an environment variable present in
+ a Container.
+ properties:
+ name:
+ description: Name of the environment variable. Must be a C_IDENTIFIER.
+ type: string
+ value:
+ description: 'Variable references $(VAR_NAME) are expanded using
+ the previously defined environment variables in the container
+ and any service environment variables. If a variable cannot
+ be resolved, the reference in the input string will be unchanged.
+ Double $$ are reduced to a single $, which allows for escaping
+ the $(VAR_NAME) syntax: i.e. "$$(VAR_NAME)" will produce the
+ string literal "$(VAR_NAME)". Escaped references will never
+ be expanded, regardless of whether the variable exists or
+ not. Defaults to "".'
+ type: string
+ valueFrom:
+ description: Source for the environment variable's value. Cannot
+ be used if value is not empty.
+ properties:
+ configMapKeyRef:
+ description: Selects a key of a ConfigMap.
+ properties:
+ key:
+ description: The key to select.
+ type: string
+ name:
+ description: 'Name of the referent. More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names
+ TODO: Add other useful fields. apiVersion, kind, uid?'
+ type: string
+ optional:
+ description: Specify whether the ConfigMap or its key
+ must be defined
+ type: boolean
+ required:
+ - key
+ type: object
+ x-kubernetes-map-type: atomic
+ fieldRef:
+ description: 'Selects a field of the pod: supports metadata.name,
+ metadata.namespace, `metadata.labels['''']`, `metadata.annotations['''']`,
+ spec.nodeName, spec.serviceAccountName, status.hostIP,
+ status.podIP, status.podIPs.'
+ properties:
+ apiVersion:
+ description: Version of the schema the FieldPath is
+ written in terms of, defaults to "v1".
+ type: string
+ fieldPath:
+ description: Path of the field to select in the specified
+ API version.
+ type: string
+ required:
+ - fieldPath
+ type: object
+ x-kubernetes-map-type: atomic
+ resourceFieldRef:
+ description: 'Selects a resource of the container: only
+ resources limits and requests (limits.cpu, limits.memory,
+ limits.ephemeral-storage, requests.cpu, requests.memory
+ and requests.ephemeral-storage) are currently supported.'
+ properties:
+ containerName:
+ description: 'Container name: required for volumes,
+ optional for env vars'
+ type: string
+ divisor:
+ anyOf:
+ - type: integer
+ - type: string
+ description: Specifies the output format of the exposed
+ resources, defaults to "1"
+ pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$
+ x-kubernetes-int-or-string: true
+ resource:
+ description: 'Required: resource to select'
+ type: string
+ required:
+ - resource
+ type: object
+ x-kubernetes-map-type: atomic
+ secretKeyRef:
+ description: Selects a key of a secret in the pod's namespace
+ properties:
+ key:
+ description: The key of the secret to select from. Must
+ be a valid secret key.
+ type: string
+ name:
+ description: 'Name of the referent. More info: https://kubernetes.io/docs/concepts/overview/working-with-objects/names/#names
+ TODO: Add other useful fields. apiVersion, kind, uid?'
+ type: string
+ optional:
+ description: Specify whether the Secret or its key must
+ be defined
+ type: boolean
+ required:
+ - key
+ type: object
+ x-kubernetes-map-type: atomic
+ type: object
+ required:
+ - name
+ type: object
+ type: array
+ x-kubernetes-preserve-unknown-fields: true
+ prepareDataConfig:
+ description: configuration for the action of "prepareData" phase,
+ including the persistent volume claims that need to be restored
+ and scheduling strategy of temporary recovery pod.
+ properties:
+ dataSourceRef:
+ description: dataSourceRef describes the configuration when using
+ `persistentVolumeClaim.spec.dataSourceRef` method for restoring.
+ it describes the source volume of the backup targetVolumes and
+ how to mount path in the restoring container.
+ properties:
+ mountPath:
+ description: mountPath path within the restoring container
+ at which the volume should be mounted.
+ type: string
+ volumeSource:
+ description: volumeSource describes the volume will be restored
+ from the specified volume of the backup targetVolumes. required
+ if the backup uses volume snapshot.
+ type: string
+ type: object
+ x-kubernetes-validations:
+ - message: at least one exists for volumeSource and mountPath.
+ rule: self.volumeSource != '' || self.mountPath !=''
+ - message: forbidden to update spec.prepareDataConfig.dataSourceRef
+ rule: self == oldSelf
+ schedulingSpec:
+ description: scheduling spec for restoring pod.
+ properties:
+ affinity:
+ description: affinity is a group of affinity scheduling rules.
+ refer to https://kubernetes.io/docs/concepts/configuration/assign-pod-node/
+ properties:
+ nodeAffinity:
+ description: Describes node affinity scheduling rules
+ for the pod.
+ properties:
+ preferredDuringSchedulingIgnoredDuringExecution:
+ description: The scheduler will prefer to schedule
+ pods to nodes that satisfy the affinity expressions
+ specified by this field, but it may choose a node
+ that violates one or more of the expressions. The
+ node that is most preferred is the one with the
+ greatest sum of weights, i.e. for each node that
+ meets all of the scheduling requirements (resource
+ request, requiredDuringScheduling affinity expressions,
+ etc.), compute a sum by iterating through the elements
+ of this field and adding "weight" to the sum if
+ the node matches the corresponding matchExpressions;
+ the node(s) with the highest sum are the most preferred.
+ items:
+ description: An empty preferred scheduling term
+ matches all objects with implicit weight 0 (i.e.
+ it's a no-op). A null preferred scheduling term
+ matches no objects (i.e. is also a no-op).
+ properties:
+ preference:
+ description: A node selector term, associated
+ with the corresponding weight.
+ properties:
+ matchExpressions:
+ description: A list of node selector requirements
+ by node's labels.
+ items:
+ description: A node selector requirement
+ is a selector that contains values,
+ a key, and an operator that relates
+ the key and values.
+ properties:
+ key:
+ description: The label key that the
+ selector applies to.
+ type: string
+ operator:
+ description: Represents a key's relationship
+ to a set of values. Valid operators
+ are In, NotIn, Exists, DoesNotExist.
+ Gt, and Lt.
+ type: string
+ values:
+ description: An array of string values.
+ If the operator is In or NotIn,
+ the values array must be non-empty.
+ If the operator is Exists or DoesNotExist,
+ the values array must be empty.
+ If the operator is Gt or Lt, the
+ values array must have a single
+ element, which will be interpreted
+ as an integer. This array is replaced
+ during a strategic merge patch.
+ items:
+ type: string
+ type: array
+ required:
+ - key
+ - operator
+ type: object
+ type: array
+ matchFields:
+ description: A list of node selector requirements
+ by node's fields.
+ items:
+ description: A node selector requirement
+ is a selector that contains values,
+ a key, and an operator that relates
+ the key and values.
+ properties:
+ key:
+ description: The label key that the
+ selector applies to.
+ type: string
+ operator:
+ description: Represents a key's relationship
+ to a set of values. Valid operators
+ are In, NotIn, Exists, DoesNotExist.
+ Gt, and Lt.
+ type: string
+ values:
+ description: An array of string values.
+ If the operator is In or NotIn,
+ the values array must be non-empty.
+ If the operator is Exists or DoesNotExist,
+ the values array must be empty.
+ If the operator is Gt or Lt, the
+ values array must have a single
+ element, which will be interpreted
+ as an integer. This array is replaced
+ during a strategic merge patch.
+ items:
+ type: string
+ type: array
+ required:
+ - key
+ - operator
+ type: object
+ type: array
+ type: object
+ x-kubernetes-map-type: atomic
+ weight:
+ description: Weight associated with matching
+ the corresponding nodeSelectorTerm, in the
+ range 1-100.
+ format: int32
+ type: integer
+ required:
+ - preference
+ - weight
+ type: object
+ type: array
+ requiredDuringSchedulingIgnoredDuringExecution:
+ description: If the affinity requirements specified
+ by this field are not met at scheduling time, the
+ pod will not be scheduled onto the node. If the
+ affinity requirements specified by this field cease
+ to be met at some point during pod execution (e.g.
+ due to an update), the system may or may not try
+ to eventually evict the pod from its node.
+ properties:
+ nodeSelectorTerms:
+ description: Required. A list of node selector
+ terms. The terms are ORed.
+ items:
+ description: A null or empty node selector term
+ matches no objects. The requirements of them
+ are ANDed. The TopologySelectorTerm type implements
+ a subset of the NodeSelectorTerm.
+ properties:
+ matchExpressions:
+ description: A list of node selector requirements
+ by node's labels.
+ items:
+ description: A node selector requirement
+ is a selector that contains values,
+ a key, and an operator that relates
+ the key and values.
+ properties:
+ key:
+ description: The label key that the
+ selector applies to.
+ type: string
+ operator:
+ description: Represents a key's relationship
+ to a set of values. Valid operators
+ are In, NotIn, Exists, DoesNotExist.
+ Gt, and Lt.
+ type: string
+ values:
+ description: An array of string values.
+ If the operator is In or NotIn,
+ the values array must be non-empty.
+ If the operator is Exists or DoesNotExist,
+ the values array must be empty.
+ If the operator is Gt or Lt, the
+ values array must have a single
+ element, which will be interpreted
+ as an integer. This array is replaced
+ during a strategic merge patch.
+ items:
+ type: string
+ type: array
+ required:
+ - key
+ - operator
+ type: object
+ type: array
+ matchFields:
+ description: A list of node selector requirements
+ by node's fields.
+ items:
+ description: A node selector requirement
+ is a selector that contains values,
+ a key, and an operator that relates
+ the key and values.
+ properties:
+ key:
+ description: The label key that the
+ selector applies to.
+ type: string
+ operator:
+ description: Represents a key's relationship
+ to a set of values. Valid operators
+ are In, NotIn, Exists, DoesNotExist.
+ Gt, and Lt.
+ type: string
+ values:
+ description: An array of string values.
+ If the operator is In or NotIn,
+ the values array must be non-empty.
+ If the operator is Exists or DoesNotExist,
+ the values array must be empty.
+ If the operator is Gt or Lt, the
+ values array must have a single
+ element, which will be interpreted
+ as an integer. This array is replaced
+ during a strategic merge patch.
+ items:
+ type: string
+ type: array
+ required:
+ - key
+ - operator
+ type: object
+ type: array
+ type: object
+ x-kubernetes-map-type: atomic
+ type: array
+ required:
+ - nodeSelectorTerms
+ type: object
+ x-kubernetes-map-type: atomic
+ type: object
+ podAffinity:
+ description: Describes pod affinity scheduling rules (e.g.
+ co-locate this pod in the same node, zone, etc. as some
+ other pod(s)).
+ properties:
+ preferredDuringSchedulingIgnoredDuringExecution:
+ description: The scheduler will prefer to schedule
+ pods to nodes that satisfy the affinity expressions
+ specified by this field, but it may choose a node
+ that violates one or more of the expressions. The
+ node that is most preferred is the one with the
+ greatest sum of weights, i.e. for each node that
+ meets all of the scheduling requirements (resource
+ request, requiredDuringScheduling affinity expressions,
+ etc.), compute a sum by iterating through the elements
+ of this field and adding "weight" to the sum if
+ the node has pods which matches the corresponding
+ podAffinityTerm; the node(s) with the highest sum
+ are the most preferred.
+ items:
+ description: The weights of all of the matched WeightedPodAffinityTerm
+ fields are added per-node to find the most preferred
+ node(s)
+ properties:
+ podAffinityTerm:
+ description: Required. A pod affinity term,
+ associated with the corresponding weight.
+ properties:
+ labelSelector:
+ description: A label query over a set of
+ resources, in this case pods.
+ properties:
+ matchExpressions:
+ description: matchExpressions is a list
+ of label selector requirements. The
+ requirements are ANDed.
+ items:
+ description: A label selector requirement
+ is a selector that contains values,
+ a key, and an operator that relates
+ the key and values.
+ properties:
+ key:
+ description: key is the label
+ key that the selector applies
+ to.
+ type: string
+ operator:
+ description: operator represents
+ a key's relationship to a set
+ of values. Valid operators are
+ In, NotIn, Exists and DoesNotExist.
+ type: string
+ values:
+ description: values is an array
+ of string values. If the operator
+ is In or NotIn, the values array
+ must be non-empty. If the operator
+ is Exists or DoesNotExist, the
+ values array must be empty.
+ This array is replaced during
+ a strategic merge patch.
+ items:
+ type: string
+ type: array
+ required:
+ - key
+ - operator
+ type: object
+ type: array
+ matchLabels:
+ additionalProperties:
+ type: string
+ description: matchLabels is a map of
+ {key,value} pairs. A single {key,value}
+ in the matchLabels map is equivalent
+ to an element of matchExpressions,
+ whose key field is "key", the operator
+ is "In", and the values array contains
+ only "value". The requirements are
+ ANDed.
+ type: object
+ type: object
+ x-kubernetes-map-type: atomic
+ namespaceSelector:
+ description: A label query over the set
+ of namespaces that the term applies to.
+ The term is applied to the union of the
+ namespaces selected by this field and
+ the ones listed in the namespaces field.
+ null selector and null or empty namespaces
+ list means "this pod's namespace". An
+ empty selector ({}) matches all namespaces.
+ properties:
+ matchExpressions:
+ description: matchExpressions is a list
+ of label selector requirements. The
+ requirements are ANDed.
+ items:
+ description: A label selector requirement
+ is a selector that contains values,
+ a key, and an operator that relates
+ the key and values.
+ properties:
+ key:
+ description: key is the label
+ key that the selector applies
+ to.
+ type: string
+ operator:
+ description: operator represents
+ a key's relationship to a set
+ of values. Valid operators are
+ In, NotIn, Exists and DoesNotExist.
+ type: string
+ values:
+ description: values is an array
+ of string values. If the operator
+ is In or NotIn, the values array
+ must be non-empty. If the operator
+ is Exists or DoesNotExist, the
+ values array must be empty.
+ This array is replaced during
+ a strategic merge patch.
+ items:
+ type: string
+ type: array
+ required:
+ - key
+ - operator
+ type: object
+ type: array
+ matchLabels:
+ additionalProperties:
+ type: string
+ description: matchLabels is a map of
+ {key,value} pairs. A single {key,value}
+ in the matchLabels map is equivalent
+ to an element of matchExpressions,
+ whose key field is "key", the operator
+ is "In", and the values array contains
+ only "value". The requirements are
+ ANDed.
+ type: object
+ type: object
+ x-kubernetes-map-type: atomic
+ namespaces:
+ description: namespaces specifies a static
+ list of namespace names that the term
+ applies to. The term is applied to the
+ union of the namespaces listed in this
+ field and the ones selected by namespaceSelector.
+ null or empty namespaces list and null
+ namespaceSelector means "this pod's namespace".
+ items:
+ type: string
+ type: array
+ topologyKey:
+ description: This pod should be co-located
+ (affinity) or not co-located (anti-affinity)
+ with the pods matching the labelSelector
+ in the specified namespaces, where co-located
+ is defined as running on a node whose
+ value of the label with key topologyKey
+ matches that of any node on which any
+ of the selected pods is running. Empty
+ topologyKey is not allowed.
+ type: string
+ required:
+ - topologyKey
+ type: object
+ weight:
+ description: weight associated with matching
+ the corresponding podAffinityTerm, in the
+ range 1-100.
+ format: int32
+ type: integer
+ required:
+ - podAffinityTerm
+ - weight
+ type: object
+ type: array
+ requiredDuringSchedulingIgnoredDuringExecution:
+ description: If the affinity requirements specified
+ by this field are not met at scheduling time, the
+ pod will not be scheduled onto the node. If the
+ affinity requirements specified by this field cease
+ to be met at some point during pod execution (e.g.
+ due to a pod label update), the system may or may
+ not try to eventually evict the pod from its node.
+ When there are multiple elements, the lists of nodes
+ corresponding to each podAffinityTerm are intersected,
+ i.e. all terms must be satisfied.
+ items:
+ description: Defines a set of pods (namely those
+ matching the labelSelector relative to the given
+ namespace(s)) that this pod should be co-located
+ (affinity) or not co-located (anti-affinity) with,
+ where co-located is defined as running on a node
+ whose value of the label with key
+ matches that of any node on which a pod of the
+ set of pods is running
+ properties:
+ labelSelector:
+ description: A label query over a set of resources,
+ in this case pods.
+ properties:
+ matchExpressions:
+ description: matchExpressions is a list
+ of label selector requirements. The requirements
+ are ANDed.
+ items:
+ description: A label selector requirement
+ is a selector that contains values,
+ a key, and an operator that relates
+ the key and values.
+ properties:
+ key:
+ description: key is the label key
+ that the selector applies to.
+ type: string
+ operator:
+ description: operator represents a
+ key's relationship to a set of values.
+ Valid operators are In, NotIn, Exists
+ and DoesNotExist.
+ type: string
+ values:
+ description: values is an array of
+ string values. If the operator is
+ In or NotIn, the values array must
+ be non-empty. If the operator is
+ Exists or DoesNotExist, the values
+ array must be empty. This array
+ is replaced during a strategic merge
+ patch.
+ items:
+ type: string
+ type: array
+ required:
+ - key
+ - operator
+ type: object
+ type: array
+ matchLabels:
+ additionalProperties:
+ type: string
+ description: matchLabels is a map of {key,value}
+ pairs. A single {key,value} in the matchLabels
+ map is equivalent to an element of matchExpressions,
+ whose key field is "key", the operator
+ is "In", and the values array contains
+ only "value". The requirements are ANDed.
+ type: object
+ type: object
+ x-kubernetes-map-type: atomic
+ namespaceSelector:
+ description: A label query over the set of namespaces
+ that the term applies to. The term is applied
+ to the union of the namespaces selected by
+ this field and the ones listed in the namespaces
+ field. null selector and null or empty namespaces
+ list means "this pod's namespace". An empty
+ selector ({}) matches all namespaces.
+ properties:
+ matchExpressions:
+ description: matchExpressions is a list
+ of label selector requirements. The requirements
+ are ANDed.
+ items:
+ description: A label selector requirement
+ is a selector that contains values,
+ a key, and an operator that relates
+ the key and values.
+ properties:
+ key:
+ description: key is the label key
+ that the selector applies to.
+ type: string
+ operator:
+ description: operator represents a
+ key's relationship to a set of values.
+ Valid operators are In, NotIn, Exists
+ and DoesNotExist.
+ type: string
+ values:
+ description: values is an array of
+ string values. If the operator is
+ In or NotIn, the values array must
+ be non-empty. If the operator is
+ Exists or DoesNotExist, the values
+ array must be empty. This array
+ is replaced during a strategic merge
+ patch.
+ items:
+ type: string
+ type: array
+ required:
+ - key
+ - operator
+ type: object
+ type: array
+ matchLabels:
+ additionalProperties:
+ type: string
+ description: matchLabels is a map of {key,value}
+ pairs. A single {key,value} in the matchLabels
+ map is equivalent to an element of matchExpressions,
+ whose key field is "key", the operator
+ is "In", and the values array contains
+ only "value". The requirements are ANDed.
+ type: object
+ type: object
+ x-kubernetes-map-type: atomic
+ namespaces:
+ description: namespaces specifies a static list
+ of namespace names that the term applies to.
+ The term is applied to the union of the namespaces
+ listed in this field and the ones selected
+ by namespaceSelector. null or empty namespaces
+ list and null namespaceSelector means "this
+ pod's namespace".
+ items:
+ type: string
+ type: array
+ topologyKey:
+ description: This pod should be co-located (affinity)
+ or not co-located (anti-affinity) with the
+ pods matching the labelSelector in the specified
+ namespaces, where co-located is defined as
+ running on a node whose value of the label
+ with key topologyKey matches that of any node
+ on which any of the selected pods is running.
+ Empty topologyKey is not allowed.
+ type: string
+ required:
+ - topologyKey
+ type: object
+ type: array
+ type: object
+ podAntiAffinity:
+ description: Describes pod anti-affinity scheduling rules
+ (e.g. avoid putting this pod in the same node, zone,
+ etc. as some other pod(s)).
+ properties:
+ preferredDuringSchedulingIgnoredDuringExecution:
+ description: The scheduler will prefer to schedule
+ pods to nodes that satisfy the anti-affinity expressions
+ specified by this field, but it may choose a node
+ that violates one or more of the expressions. The
+ node that is most preferred is the one with the
+ greatest sum of weights, i.e. for each node that
+ meets all of the scheduling requirements (resource
+ request, requiredDuringScheduling anti-affinity
+ expressions, etc.), compute a sum by iterating through
+ the elements of this field and adding "weight" to
+ the sum if the node has pods which matches the corresponding
+ podAffinityTerm; the node(s) with the highest sum
+ are the most preferred.
+ items:
+ description: The weights of all of the matched WeightedPodAffinityTerm
+ fields are added per-node to find the most preferred
+ node(s)
+ properties:
+ podAffinityTerm:
+ description: Required. A pod affinity term,
+ associated with the corresponding weight.
+ properties:
+ labelSelector:
+ description: A label query over a set of
+ resources, in this case pods.
+ properties:
+ matchExpressions:
+ description: matchExpressions is a list
+ of label selector requirements. The
+ requirements are ANDed.
+ items:
+ description: A label selector requirement
+ is a selector that contains values,
+ a key, and an operator that relates
+ the key and values.
+ properties:
+ key:
+ description: key is the label
+ key that the selector applies
+ to.
+ type: string
+ operator:
+ description: operator represents
+ a key's relationship to a set
+ of values. Valid operators are
+ In, NotIn, Exists and DoesNotExist.
+ type: string
+ values:
+ description: values is an array
+ of string values. If the operator
+ is In or NotIn, the values array
+ must be non-empty. If the operator
+ is Exists or DoesNotExist, the
+ values array must be empty.
+ This array is replaced during
+ a strategic merge patch.
+ items:
+ type: string
+ type: array
+ required:
+ - key
+ - operator
+ type: object
+ type: array
+ matchLabels:
+ additionalProperties:
+ type: string
+ description: matchLabels is a map of
+ {key,value} pairs. A single {key,value}
+ in the matchLabels map is equivalent
+ to an element of matchExpressions,
+ whose key field is "key", the operator
+ is "In", and the values array contains
+ only "value". The requirements are
+ ANDed.
+ type: object
+ type: object
+ x-kubernetes-map-type: atomic
+ namespaceSelector:
+ description: A label query over the set
+ of namespaces that the term applies to.
+ The term is applied to the union of the
+ namespaces selected by this field and
+ the ones listed in the namespaces field.
+ null selector and null or empty namespaces
+ list means "this pod's namespace". An
+ empty selector ({}) matches all namespaces.
+ properties:
+ matchExpressions:
+ description: matchExpressions is a list
+ of label selector requirements. The
+ requirements are ANDed.
+ items:
+ description: A label selector requirement
+ is a selector that contains values,
+ a key, and an operator that relates
+ the key and values.
+ properties:
+ key:
+ description: key is the label
+ key that the selector applies
+ to.
+ type: string
+ operator:
+ description: operator represents
+ a key's relationship to a set
+ of values. Valid operators are
+ In, NotIn, Exists and DoesNotExist.
+ type: string
+ values:
+ description: values is an array
+ of string values. If the operator
+ is In or NotIn, the values array
+ must be non-empty. If the operator
+ is Exists or DoesNotExist, the
+ values array must be empty.
+ This array is replaced during
+ a strategic merge patch.
+ items:
+ type: string
+ type: array
+ required:
+ - key
+ - operator
+ type: object
+ type: array
+ matchLabels:
+ additionalProperties:
+ type: string
+ description: matchLabels is a map of
+ {key,value} pairs. A single {key,value}
+ in the matchLabels map is equivalent
+ to an element of matchExpressions,
+ whose key field is "key", the operator
+ is "In", and the values array contains
+ only "value". The requirements are
+ ANDed.
+ type: object
+ type: object
+ x-kubernetes-map-type: atomic
+ namespaces:
+ description: namespaces specifies a static
+ list of namespace names that the term
+ applies to. The term is applied to the
+ union of the namespaces listed in this
+ field and the ones selected by namespaceSelector.
+ null or empty namespaces list and null
+ namespaceSelector means "this pod's namespace".
+ items:
+ type: string
+ type: array
+ topologyKey:
+ description: This pod should be co-located
+ (affinity) or not co-located (anti-affinity)
+ with the pods matching the labelSelector
+ in the specified namespaces, where co-located
+ is defined as running on a node whose
+ value of the label with key topologyKey
+ matches that of any node on which any
+ of the selected pods is running. Empty
+ topologyKey is not allowed.
+ type: string
+ required:
+ - topologyKey
+ type: object
+ weight:
+ description: weight associated with matching
+ the corresponding podAffinityTerm, in the
+ range 1-100.
+ format: int32
+ type: integer
+ required:
+ - podAffinityTerm
+ - weight
+ type: object
+ type: array
+ requiredDuringSchedulingIgnoredDuringExecution:
+ description: If the anti-affinity requirements specified
+ by this field are not met at scheduling time, the
+ pod will not be scheduled onto the node. If the
+ anti-affinity requirements specified by this field
+ cease to be met at some point during pod execution
+ (e.g. due to a pod label update), the system may
+ or may not try to eventually evict the pod from
+ its node. When there are multiple elements, the
+ lists of nodes corresponding to each podAffinityTerm
+ are intersected, i.e. all terms must be satisfied.
+ items:
+ description: Defines a set of pods (namely those
+ matching the labelSelector relative to the given
+ namespace(s)) that this pod should be co-located
+ (affinity) or not co-located (anti-affinity) with,
+ where co-located is defined as running on a node
+ whose value of the label with key
+ matches that of any node on which a pod of the
+ set of pods is running
+ properties:
+ labelSelector:
+ description: A label query over a set of resources,
+ in this case pods.
+ properties:
+ matchExpressions:
+ description: matchExpressions is a list
+ of label selector requirements. The requirements
+ are ANDed.
+ items:
+ description: A label selector requirement
+ is a selector that contains values,
+ a key, and an operator that relates
+ the key and values.
+ properties:
+ key:
+ description: key is the label key
+ that the selector applies to.
+ type: string
+ operator:
+ description: operator represents a
+ key's relationship to a set of values.
+ Valid operators are In, NotIn, Exists
+ and DoesNotExist.
+ type: string
+ values:
+ description: values is an array of
+ string values. If the operator is
+ In or NotIn, the values array must
+ be non-empty. If the operator is
+ Exists or DoesNotExist, the values
+ array must be empty. This array
+ is replaced during a strategic merge
+ patch.
+ items:
+ type: string
+ type: array
+ required:
+ - key
+ - operator
+ type: object
+ type: array
+ matchLabels:
+ additionalProperties:
+ type: string
+ description: matchLabels is a map of {key,value}
+ pairs. A single {key,value} in the matchLabels
+ map is equivalent to an element of matchExpressions,
+ whose key field is "key", the operator
+ is "In", and the values array contains
+ only "value". The requirements are ANDed.
+ type: object
+ type: object
+ x-kubernetes-map-type: atomic
+ namespaceSelector:
+ description: A label query over the set of namespaces
+ that the term applies to. The term is applied
+ to the union of the namespaces selected by
+ this field and the ones listed in the namespaces
+ field. null selector and null or empty namespaces
+ list means "this pod's namespace". An empty
+ selector ({}) matches all namespaces.
+ properties:
+ matchExpressions:
+ description: matchExpressions is a list
+ of label selector requirements. The requirements
+ are ANDed.
+ items:
+ description: A label selector requirement
+ is a selector that contains values,
+ a key, and an operator that relates
+ the key and values.
+ properties:
+ key:
+ description: key is the label key
+ that the selector applies to.
+ type: string
+ operator:
+ description: operator represents a
+ key's relationship to a set of values.
+ Valid operators are In, NotIn, Exists
+ and DoesNotExist.
+ type: string
+ values:
+ description: values is an array of
+ string values. If the operator is
+ In or NotIn, the values array must
+ be non-empty. If the operator is
+ Exists or DoesNotExist, the values
+ array must be empty. This array
+ is replaced during a strategic merge
+ patch.
+ items:
+ type: string
+ type: array
+ required:
+ - key
+ - operator
+ type: object
+ type: array
+ matchLabels:
+ additionalProperties:
+ type: string
+ description: matchLabels is a map of {key,value}
+ pairs. A single {key,value} in the matchLabels
+ map is equivalent to an element of matchExpressions,
+ whose key field is "key", the operator
+ is "In", and the values array contains
+ only "value". The requirements are ANDed.
+ type: object
+ type: object
+ x-kubernetes-map-type: atomic
+ namespaces:
+ description: namespaces specifies a static list
+ of namespace names that the term applies to.
+ The term is applied to the union of the namespaces
+ listed in this field and the ones selected
+ by namespaceSelector. null or empty namespaces
+ list and null namespaceSelector means "this
+ pod's namespace".
+ items:
+ type: string
+ type: array
+ topologyKey:
+ description: This pod should be co-located (affinity)
+ or not co-located (anti-affinity) with the
+ pods matching the labelSelector in the specified
+ namespaces, where co-located is defined as
+ running on a node whose value of the label
+ with key topologyKey matches that of any node
+ on which any of the selected pods is running.
+ Empty topologyKey is not allowed.
+ type: string
+ required:
+ - topologyKey
+ type: object
+ type: array
+ type: object
+ type: object
+ nodeName:
+ description: nodeName is a request to schedule this pod onto
+ a specific node. If it is non-empty, the scheduler simply
+ schedules this pod onto that node, assuming that it fits
+ resource requirements.
+ type: string
+ nodeSelector:
+ additionalProperties:
+ type: string
+ description: 'nodeSelector is a selector which must be true
+ for the pod to fit on a node. Selector which must match
+ a node''s labels for the pod to be scheduled on that node.
+ More info: https://kubernetes.io/docs/concepts/configuration/assign-pod-node/'
+ type: object
+ x-kubernetes-map-type: atomic
+ schedulerName:
+ description: If specified, the pod will be dispatched by specified
+ scheduler. If not specified, the pod will be dispatched
+ by default scheduler.
+ type: string
+ tolerations:
+ description: the restoring pod's tolerations.
+ items:
+ description: The pod this Toleration is attached to tolerates
+ any taint that matches the triple using
+ the matching operator .
+ properties:
+ effect:
+ description: Effect indicates the taint effect to match.
+ Empty means match all taint effects. When specified,
+ allowed values are NoSchedule, PreferNoSchedule and
+ NoExecute.
+ type: string
+ key:
+ description: Key is the taint key that the toleration
+ applies to. Empty means match all taint keys. If the
+ key is empty, operator must be Exists; this combination
+ means to match all values and all keys.
+ type: string
+ operator:
+ description: Operator represents a key's relationship
+ to the value. Valid operators are Exists and Equal.
+ Defaults to Equal. Exists is equivalent to wildcard
+ for value, so that a pod can tolerate all taints of
+ a particular category.
+ type: string
+ tolerationSeconds:
+ description: TolerationSeconds represents the period
+ of time the toleration (which must be of effect NoExecute,
+ otherwise this field is ignored) tolerates the taint.
+ By default, it is not set, which means tolerate the
+ taint forever (do not evict). Zero and negative values
+ will be treated as 0 (evict immediately) by the system.
+ format: int64
+ type: integer
+ value:
+ description: Value is the taint value the toleration
+ matches to. If the operator is Exists, the value should
+ be empty, otherwise just a regular string.
+ type: string
+ type: object
+ type: array
+ topologySpreadConstraints:
+ description: topologySpreadConstraints describes how a group
+ of pods ought to spread across topology domains. Scheduler
+ will schedule pods in a way which abides by the constraints.
+ refer to https://kubernetes.io/docs/concepts/scheduling-eviction/topology-spread-constraints/
+ items:
+ description: TopologySpreadConstraint specifies how to spread
+ matching pods among the given topology.
+ properties:
+ labelSelector:
+ description: LabelSelector is used to find matching
+ pods. Pods that match this label selector are counted
+ to determine the number of pods in their corresponding
+ topology domain.
+ properties:
+ matchExpressions:
+ description: matchExpressions is a list of label
+ selector requirements. The requirements are ANDed.
+ items:
+ description: A label selector requirement is a
+ selector that contains values, a key, and an
+ operator that relates the key and values.
+ properties:
+ key:
+ description: key is the label key that the
+ selector applies to.
+ type: string
+ operator:
+ description: operator represents a key's relationship
+ to a set of values. Valid operators are
+ In, NotIn, Exists and DoesNotExist.
+ type: string
+ values:
+ description: values is an array of string
+ values. If the operator is In or NotIn,
+ the values array must be non-empty. If the
+ operator is Exists or DoesNotExist, the
+ values array must be empty. This array is
+ replaced during a strategic merge patch.
+ items:
+ type: string
+ type: array
+ required:
+ - key
+ - operator
+ type: object
+ type: array
+ matchLabels:
+ additionalProperties:
+ type: string
+ description: matchLabels is a map of {key,value}
+ pairs. A single {key,value} in the matchLabels
+ map is equivalent to an element of matchExpressions,
+ whose key field is "key", the operator is "In",
+ and the values array contains only "value". The
+ requirements are ANDed.
+ type: object
+ type: object
+ x-kubernetes-map-type: atomic
+ matchLabelKeys:
+ description: "MatchLabelKeys is a set of pod label keys
+ to select the pods over which spreading will be calculated.
+ The keys are used to lookup values from the incoming
+ pod labels, those key-value labels are ANDed with
+ labelSelector to select the group of existing pods
+ over which spreading will be calculated for the incoming
+ pod. The same key is forbidden to exist in both MatchLabelKeys
+ and LabelSelector. MatchLabelKeys cannot be set when
+ LabelSelector isn't set. Keys that don't exist in
+ the incoming pod labels will be ignored. A null or
+ empty list means only match against labelSelector.
+ \n This is a beta field and requires the MatchLabelKeysInPodTopologySpread
+ feature gate to be enabled (enabled by default)."
+ items:
+ type: string
+ type: array
+ x-kubernetes-list-type: atomic
+ maxSkew:
+ description: 'MaxSkew describes the degree to which
+ pods may be unevenly distributed. When `whenUnsatisfiable=DoNotSchedule`,
+ it is the maximum permitted difference between the
+ number of matching pods in the target topology and
+ the global minimum. The global minimum is the minimum
+ number of matching pods in an eligible domain or zero
+ if the number of eligible domains is less than MinDomains.
+ For example, in a 3-zone cluster, MaxSkew is set to
+ 1, and pods with the same labelSelector spread as
+ 2/2/1: In this case, the global minimum is 1. | zone1
+ | zone2 | zone3 | | P P | P P | P | - if MaxSkew
+ is 1, incoming pod can only be scheduled to zone3
+ to become 2/2/2; scheduling it onto zone1(zone2) would
+ make the ActualSkew(3-1) on zone1(zone2) violate MaxSkew(1).
+ - if MaxSkew is 2, incoming pod can be scheduled onto
+ any zone. When `whenUnsatisfiable=ScheduleAnyway`,
+ it is used to give higher precedence to topologies
+ that satisfy it. It''s a required field. Default value
+ is 1 and 0 is not allowed.'
+ format: int32
+ type: integer
+ minDomains:
+ description: "MinDomains indicates a minimum number
+ of eligible domains. When the number of eligible domains
+ with matching topology keys is less than minDomains,
+ Pod Topology Spread treats \"global minimum\" as 0,
+ and then the calculation of Skew is performed. And
+ when the number of eligible domains with matching
+ topology keys equals or greater than minDomains, this
+ value has no effect on scheduling. As a result, when
+ the number of eligible domains is less than minDomains,
+ scheduler won't schedule more than maxSkew Pods to
+ those domains. If value is nil, the constraint behaves
+ as if MinDomains is equal to 1. Valid values are integers
+ greater than 0. When value is not nil, WhenUnsatisfiable
+ must be DoNotSchedule. \n For example, in a 3-zone
+ cluster, MaxSkew is set to 2, MinDomains is set to
+ 5 and pods with the same labelSelector spread as 2/2/2:
+ | zone1 | zone2 | zone3 | | P P | P P | P P |
+ The number of domains is less than 5(MinDomains),
+ so \"global minimum\" is treated as 0. In this situation,
+ new pod with the same labelSelector cannot be scheduled,
+ because computed skew will be 3(3 - 0) if new Pod
+ is scheduled to any of the three zones, it will violate
+ MaxSkew. \n This is a beta field and requires the
+ MinDomainsInPodTopologySpread feature gate to be enabled
+ (enabled by default)."
+ format: int32
+ type: integer
+ nodeAffinityPolicy:
+ description: "NodeAffinityPolicy indicates how we will
+ treat Pod's nodeAffinity/nodeSelector when calculating
+ pod topology spread skew. Options are: - Honor: only
+ nodes matching nodeAffinity/nodeSelector are included
+ in the calculations. - Ignore: nodeAffinity/nodeSelector
+ are ignored. All nodes are included in the calculations.
+ \n If this value is nil, the behavior is equivalent
+ to the Honor policy. This is a beta-level feature
+ default enabled by the NodeInclusionPolicyInPodTopologySpread
+ feature flag."
+ type: string
+ nodeTaintsPolicy:
+ description: "NodeTaintsPolicy indicates how we will
+ treat node taints when calculating pod topology spread
+ skew. Options are: - Honor: nodes without taints,
+ along with tainted nodes for which the incoming pod
+ has a toleration, are included. - Ignore: node taints
+ are ignored. All nodes are included. \n If this value
+ is nil, the behavior is equivalent to the Ignore policy.
+ This is a beta-level feature default enabled by the
+ NodeInclusionPolicyInPodTopologySpread feature flag."
+ type: string
+ topologyKey:
+ description: TopologyKey is the key of node labels.
+ Nodes that have a label with this key and identical
+ values are considered to be in the same topology.
+ We consider each as a "bucket", and try
+ to put balanced number of pods into each bucket. We
+ define a domain as a particular instance of a topology.
+ Also, we define an eligible domain as a domain whose
+ nodes meet the requirements of nodeAffinityPolicy
+ and nodeTaintsPolicy. e.g. If TopologyKey is "kubernetes.io/hostname",
+ each Node is a domain of that topology. And, if TopologyKey
+ is "topology.kubernetes.io/zone", each zone is a domain
+ of that topology. It's a required field.
+ type: string
+ whenUnsatisfiable:
+ description: 'WhenUnsatisfiable indicates how to deal
+ with a pod if it doesn''t satisfy the spread constraint.
+ - DoNotSchedule (default) tells the scheduler not
+ to schedule it. - ScheduleAnyway tells the scheduler
+ to schedule the pod in any location, but giving higher
+ precedence to topologies that would help reduce the
+ skew. A constraint is considered "Unsatisfiable" for
+ an incoming pod if and only if every possible node
+ assignment for that pod would violate "MaxSkew" on
+ some topology. For example, in a 3-zone cluster, MaxSkew
+ is set to 1, and pods with the same labelSelector
+ spread as 3/1/1: | zone1 | zone2 | zone3 | | P P P
+ | P | P | If WhenUnsatisfiable is set to DoNotSchedule,
+ incoming pod can only be scheduled to zone2(zone3)
+ to become 3/2/1(3/1/2) as ActualSkew(2-1) on zone2(zone3)
+ satisfies MaxSkew(1). In other words, the cluster
+ can still be imbalanced, but scheduler won''t make
+ it *more* imbalanced. It''s a required field.'
+ type: string
+ required:
+ - maxSkew
+ - topologyKey
+ - whenUnsatisfiable
+ type: object
+ type: array
+ type: object
+ x-kubernetes-validations:
+ - message: forbidden to update spec.prepareDataConfig.schedulingSpec
+ rule: self == oldSelf
+ volumeClaimManagementPolicy:
+ default: Parallel
+ description: 'VolumeClaimManagementPolicy defines recovery strategy
+ for persistent volume claim. supported policies are as follows:
+ 1. Parallel: parallel recovery of persistent volume claim. 2.
+ Serial: restore the persistent volume claim in sequence, and
+ wait until the previous persistent volume claim is restored
+ before restoring a new one.'
+ enum:
+ - Parallel
+ - Serial
+ type: string
+ volumeClaims:
+ description: volumeClaims defines the persistent Volume claims
+ that need to be restored and mount them together into the restore
+ job. these persistent Volume claims will be created if not exist.
+ items:
+ properties:
+ metadata:
+ description: 'Standard object''s metadata. More info: https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#metadata'
+ properties:
+ annotations:
+ additionalProperties:
+ type: string
+ type: object
+ finalizers:
+ items:
+ type: string
+ type: array
+ labels:
+ additionalProperties:
+ type: string
+ type: object
+ name:
+ type: string
+ namespace:
+ type: string
+ type: object
+ mountPath:
+ description: mountPath path within the restoring container
+ at which the volume should be mounted.
+ type: string
+ volumeClaimSpec:
+ description: volumeClaimSpec defines the desired characteristics
+ of a persistent volume claim.
+ properties:
+ accessModes:
+ description: 'accessModes contains the desired access
+ modes the volume should have. More info: https://kubernetes.io/docs/concepts/storage/persistent-volumes#access-modes-1'
+ items:
+ type: string
+ type: array
+ dataSource:
+ description: 'dataSource field can be used to specify
+ either: * An existing VolumeSnapshot object (snapshot.storage.k8s.io/VolumeSnapshot)
+ * An existing PVC (PersistentVolumeClaim) If the provisioner
+ or an external controller can support the specified
+ data source, it will create a new volume based on
+ the contents of the specified data source. When the
+ AnyVolumeDataSource feature gate is enabled, dataSource
+ contents will be copied to dataSourceRef, and dataSourceRef
+ contents will be copied to dataSource when dataSourceRef.namespace
+ is not specified. If the namespace is specified, then
+ dataSourceRef will not be copied to dataSource.'
+ properties:
+ apiGroup:
+ description: APIGroup is the group for the resource
+ being referenced. If APIGroup is not specified,
+ the specified Kind must be in the core API group.
+ For any other third-party types, APIGroup is required.
+ type: string
+ kind:
+ description: Kind is the type of resource being
+ referenced
+ type: string
+ name:
+ description: Name is the name of resource being
+ referenced
+ type: string
+ required:
+ - kind
+ - name
+ type: object
+ x-kubernetes-map-type: atomic
+ dataSourceRef:
+ description: 'dataSourceRef specifies the object from
+ which to populate the volume with data, if a non-empty
+ volume is desired. This may be any object from a non-empty
+ API group (non core object) or a PersistentVolumeClaim
+ object. When this field is specified, volume binding
+ will only succeed if the type of the specified object
+ matches some installed volume populator or dynamic
+ provisioner. This field will replace the functionality
+ of the dataSource field and as such if both fields
+ are non-empty, they must have the same value. For
+ backwards compatibility, when namespace isn''t specified
+ in dataSourceRef, both fields (dataSource and dataSourceRef)
+ will be set to the same value automatically if one
+ of them is empty and the other is non-empty. When
+ namespace is specified in dataSourceRef, dataSource
+ isn''t set to the same value and must be empty. There
+ are three important differences between dataSource
+ and dataSourceRef: * While dataSource only allows
+ two specific types of objects, dataSourceRef allows
+ any non-core object, as well as PersistentVolumeClaim
+ objects. * While dataSource ignores disallowed values
+ (dropping them), dataSourceRef preserves all values,
+ and generates an error if a disallowed value is specified.
+ * While dataSource only allows local objects, dataSourceRef
+ allows objects in any namespaces. (Beta) Using this
+ field requires the AnyVolumeDataSource feature gate
+ to be enabled. (Alpha) Using the namespace field of
+ dataSourceRef requires the CrossNamespaceVolumeDataSource
+ feature gate to be enabled.'
+ properties:
+ apiGroup:
+ description: APIGroup is the group for the resource
+ being referenced. If APIGroup is not specified,
+ the specified Kind must be in the core API group.
+ For any other third-party types, APIGroup is required.
+ type: string
+ kind:
+ description: Kind is the type of resource being
+ referenced
+ type: string
+ name:
+ description: Name is the name of resource being
+ referenced
+ type: string
+ namespace:
+ description: Namespace is the namespace of resource
+ being referenced Note that when a namespace is
+ specified, a gateway.networking.k8s.io/ReferenceGrant
+ object is required in the referent namespace to
+ allow that namespace's owner to accept the reference.
+ See the ReferenceGrant documentation for details.
+ (Alpha) This field requires the CrossNamespaceVolumeDataSource
+ feature gate to be enabled.
+ type: string
+ required:
+ - kind
+ - name
+ type: object
+ resources:
+ description: 'resources represents the minimum resources
+ the volume should have. If RecoverVolumeExpansionFailure
+ feature is enabled users are allowed to specify resource
+ requirements that are lower than previous value but
+ must still be higher than capacity recorded in the
+ status field of the claim. More info: https://kubernetes.io/docs/concepts/storage/persistent-volumes#resources'
+ properties:
+ claims:
+ description: "Claims lists the names of resources,
+ defined in spec.resourceClaims, that are used
+ by this container. \n This is an alpha field and
+ requires enabling the DynamicResourceAllocation
+ feature gate. \n This field is immutable. It can
+ only be set for containers."
+ items:
+ description: ResourceClaim references one entry
+ in PodSpec.ResourceClaims.
+ properties:
+ name:
+ description: Name must match the name of one
+ entry in pod.spec.resourceClaims of the
+ Pod where this field is used. It makes that
+ resource available inside a container.
+ type: string
+ required:
+ - name
+ type: object
+ type: array
+ x-kubernetes-list-map-keys:
+ - name
+ x-kubernetes-list-type: map
+ limits:
+ additionalProperties:
+ anyOf:
+ - type: integer
+ - type: string
+ pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$
+ x-kubernetes-int-or-string: true
+ description: 'Limits describes the maximum amount
+ of compute resources allowed. More info: https://kubernetes.io/docs/concepts/configuration/manage-resources-containers/'
+ type: object
+ requests:
+ additionalProperties:
+ anyOf:
+ - type: integer
+ - type: string
+ pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$
+ x-kubernetes-int-or-string: true
+ description: 'Requests describes the minimum amount
+ of compute resources required. If Requests is
+ omitted for a container, it defaults to Limits
+ if that is explicitly specified, otherwise to
+ an implementation-defined value. Requests cannot
+ exceed Limits. More info: https://kubernetes.io/docs/concepts/configuration/manage-resources-containers/'
+ type: object
+ type: object
+ selector:
+ description: selector is a label query over volumes
+ to consider for binding.
+ properties:
+ matchExpressions:
+ description: matchExpressions is a list of label
+ selector requirements. The requirements are ANDed.
+ items:
+ description: A label selector requirement is a
+ selector that contains values, a key, and an
+ operator that relates the key and values.
+ properties:
+ key:
+ description: key is the label key that the
+ selector applies to.
+ type: string
+ operator:
+ description: operator represents a key's relationship
+ to a set of values. Valid operators are
+ In, NotIn, Exists and DoesNotExist.
+ type: string
+ values:
+ description: values is an array of string
+ values. If the operator is In or NotIn,
+ the values array must be non-empty. If the
+ operator is Exists or DoesNotExist, the
+ values array must be empty. This array is
+ replaced during a strategic merge patch.
+ items:
+ type: string
+ type: array
+ required:
+ - key
+ - operator
+ type: object
+ type: array
+ matchLabels:
+ additionalProperties:
+ type: string
+ description: matchLabels is a map of {key,value}
+ pairs. A single {key,value} in the matchLabels
+ map is equivalent to an element of matchExpressions,
+ whose key field is "key", the operator is "In",
+ and the values array contains only "value". The
+ requirements are ANDed.
+ type: object
+ type: object
+ x-kubernetes-map-type: atomic
+ storageClassName:
+ description: 'storageClassName is the name of the StorageClass
+ required by the claim. More info: https://kubernetes.io/docs/concepts/storage/persistent-volumes#class-1'
+ type: string
+ volumeMode:
+ description: volumeMode defines what type of volume
+ is required by the claim. Value of Filesystem is implied
+ when not included in claim spec.
+ type: string
+ volumeName:
+ description: volumeName is the binding reference to
+ the PersistentVolume backing this claim.
+ type: string
+ type: object
+ volumeSource:
+ description: volumeSource describes the volume will be restored
+ from the specified volume of the backup targetVolumes.
+ required if the backup uses volume snapshot.
+ type: string
+ required:
+ - metadata
+ - volumeClaimSpec
+ type: object
+ x-kubernetes-validations:
+ - message: at least one exists for volumeSource and mountPath.
+ rule: self.volumeSource != '' || self.mountPath !=''
+ type: array
+ x-kubernetes-validations:
+ - message: forbidden to update spec.prepareDataConfig.volumeClaims
+ rule: self == oldSelf
+ volumeClaimsTemplate:
+ description: volumeClaimsTemplate defines a template to build
+ persistent Volume claims that need to be restored. these claims
+ will be created in an orderly manner based on the number of
+ replicas or reused if already exist.
+ properties:
+ replicas:
+ description: the replicas of persistent volume claim which
+ need to be created and restored. the format of created claim
+ name is "-".
+ format: int32
+ minimum: 1
+ type: integer
+ startingIndex:
+ description: the starting index for the created persistent
+ volume claim by according to template. minimum is 0.
+ format: int32
+ minimum: 0
+ type: integer
+ templates:
+ description: templates is a list of volume claims.
+ items:
+ properties:
+ metadata:
+ description: 'Standard object''s metadata. More info:
+ https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#metadata'
+ properties:
+ annotations:
+ additionalProperties:
+ type: string
+ type: object
+ finalizers:
+ items:
+ type: string
+ type: array
+ labels:
+ additionalProperties:
+ type: string
+ type: object
+ name:
+ type: string
+ namespace:
+ type: string
+ type: object
+ mountPath:
+ description: mountPath path within the restoring container
+ at which the volume should be mounted.
+ type: string
+ volumeClaimSpec:
+ description: volumeClaimSpec defines the desired characteristics
+ of a persistent volume claim.
+ properties:
+ accessModes:
+ description: 'accessModes contains the desired access
+ modes the volume should have. More info: https://kubernetes.io/docs/concepts/storage/persistent-volumes#access-modes-1'
+ items:
+ type: string
+ type: array
+ dataSource:
+ description: 'dataSource field can be used to specify
+ either: * An existing VolumeSnapshot object (snapshot.storage.k8s.io/VolumeSnapshot)
+ * An existing PVC (PersistentVolumeClaim) If the
+ provisioner or an external controller can support
+ the specified data source, it will create a new
+ volume based on the contents of the specified
+ data source. When the AnyVolumeDataSource feature
+ gate is enabled, dataSource contents will be copied
+ to dataSourceRef, and dataSourceRef contents will
+ be copied to dataSource when dataSourceRef.namespace
+ is not specified. If the namespace is specified,
+ then dataSourceRef will not be copied to dataSource.'
+ properties:
+ apiGroup:
+ description: APIGroup is the group for the resource
+ being referenced. If APIGroup is not specified,
+ the specified Kind must be in the core API
+ group. For any other third-party types, APIGroup
+ is required.
+ type: string
+ kind:
+ description: Kind is the type of resource being
+ referenced
+ type: string
+ name:
+ description: Name is the name of resource being
+ referenced
+ type: string
+ required:
+ - kind
+ - name
+ type: object
+ x-kubernetes-map-type: atomic
+ dataSourceRef:
+ description: 'dataSourceRef specifies the object
+ from which to populate the volume with data, if
+ a non-empty volume is desired. This may be any
+ object from a non-empty API group (non core object)
+ or a PersistentVolumeClaim object. When this field
+ is specified, volume binding will only succeed
+ if the type of the specified object matches some
+ installed volume populator or dynamic provisioner.
+ This field will replace the functionality of the
+ dataSource field and as such if both fields are
+ non-empty, they must have the same value. For
+ backwards compatibility, when namespace isn''t
+ specified in dataSourceRef, both fields (dataSource
+ and dataSourceRef) will be set to the same value
+ automatically if one of them is empty and the
+ other is non-empty. When namespace is specified
+ in dataSourceRef, dataSource isn''t set to the
+ same value and must be empty. There are three
+ important differences between dataSource and dataSourceRef:
+ * While dataSource only allows two specific types
+ of objects, dataSourceRef allows any non-core
+ object, as well as PersistentVolumeClaim objects.
+ * While dataSource ignores disallowed values (dropping
+ them), dataSourceRef preserves all values, and
+ generates an error if a disallowed value is specified.
+ * While dataSource only allows local objects,
+ dataSourceRef allows objects in any namespaces.
+ (Beta) Using this field requires the AnyVolumeDataSource
+ feature gate to be enabled. (Alpha) Using the
+ namespace field of dataSourceRef requires the
+ CrossNamespaceVolumeDataSource feature gate to
+ be enabled.'
+ properties:
+ apiGroup:
+ description: APIGroup is the group for the resource
+ being referenced. If APIGroup is not specified,
+ the specified Kind must be in the core API
+ group. For any other third-party types, APIGroup
+ is required.
+ type: string
+ kind:
+ description: Kind is the type of resource being
+ referenced
+ type: string
+ name:
+ description: Name is the name of resource being
+ referenced
+ type: string
+ namespace:
+ description: Namespace is the namespace of resource
+ being referenced Note that when a namespace
+ is specified, a gateway.networking.k8s.io/ReferenceGrant
+ object is required in the referent namespace
+ to allow that namespace's owner to accept
+ the reference. See the ReferenceGrant documentation
+ for details. (Alpha) This field requires the
+ CrossNamespaceVolumeDataSource feature gate
+ to be enabled.
+ type: string
+ required:
+ - kind
+ - name
+ type: object
+ resources:
+ description: 'resources represents the minimum resources
+ the volume should have. If RecoverVolumeExpansionFailure
+ feature is enabled users are allowed to specify
+ resource requirements that are lower than previous
+ value but must still be higher than capacity recorded
+ in the status field of the claim. More info: https://kubernetes.io/docs/concepts/storage/persistent-volumes#resources'
+ properties:
+ claims:
+ description: "Claims lists the names of resources,
+ defined in spec.resourceClaims, that are used
+ by this container. \n This is an alpha field
+ and requires enabling the DynamicResourceAllocation
+ feature gate. \n This field is immutable.
+ It can only be set for containers."
+ items:
+ description: ResourceClaim references one
+ entry in PodSpec.ResourceClaims.
+ properties:
+ name:
+ description: Name must match the name
+ of one entry in pod.spec.resourceClaims
+ of the Pod where this field is used.
+ It makes that resource available inside
+ a container.
+ type: string
+ required:
+ - name
+ type: object
+ type: array
+ x-kubernetes-list-map-keys:
+ - name
+ x-kubernetes-list-type: map
+ limits:
+ additionalProperties:
+ anyOf:
+ - type: integer
+ - type: string
+ pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$
+ x-kubernetes-int-or-string: true
+ description: 'Limits describes the maximum amount
+ of compute resources allowed. More info: https://kubernetes.io/docs/concepts/configuration/manage-resources-containers/'
+ type: object
+ requests:
+ additionalProperties:
+ anyOf:
+ - type: integer
+ - type: string
+ pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$
+ x-kubernetes-int-or-string: true
+ description: 'Requests describes the minimum
+ amount of compute resources required. If Requests
+ is omitted for a container, it defaults to
+ Limits if that is explicitly specified, otherwise
+ to an implementation-defined value. Requests
+ cannot exceed Limits. More info: https://kubernetes.io/docs/concepts/configuration/manage-resources-containers/'
+ type: object
+ type: object
+ selector:
+ description: selector is a label query over volumes
+ to consider for binding.
+ properties:
+ matchExpressions:
+ description: matchExpressions is a list of label
+ selector requirements. The requirements are
+ ANDed.
+ items:
+ description: A label selector requirement
+ is a selector that contains values, a key,
+ and an operator that relates the key and
+ values.
+ properties:
+ key:
+ description: key is the label key that
+ the selector applies to.
+ type: string
+ operator:
+ description: operator represents a key's
+ relationship to a set of values. Valid
+ operators are In, NotIn, Exists and
+ DoesNotExist.
+ type: string
+ values:
+ description: values is an array of string
+ values. If the operator is In or NotIn,
+ the values array must be non-empty.
+ If the operator is Exists or DoesNotExist,
+ the values array must be empty. This
+ array is replaced during a strategic
+ merge patch.
+ items:
+ type: string
+ type: array
+ required:
+ - key
+ - operator
+ type: object
+ type: array
+ matchLabels:
+ additionalProperties:
+ type: string
+ description: matchLabels is a map of {key,value}
+ pairs. A single {key,value} in the matchLabels
+ map is equivalent to an element of matchExpressions,
+ whose key field is "key", the operator is
+ "In", and the values array contains only "value".
+ The requirements are ANDed.
+ type: object
+ type: object
+ x-kubernetes-map-type: atomic
+ storageClassName:
+ description: 'storageClassName is the name of the
+ StorageClass required by the claim. More info:
+ https://kubernetes.io/docs/concepts/storage/persistent-volumes#class-1'
+ type: string
+ volumeMode:
+ description: volumeMode defines what type of volume
+ is required by the claim. Value of Filesystem
+ is implied when not included in claim spec.
+ type: string
+ volumeName:
+ description: volumeName is the binding reference
+ to the PersistentVolume backing this claim.
+ type: string
+ type: object
+ volumeSource:
+ description: volumeSource describes the volume will
+ be restored from the specified volume of the backup
+ targetVolumes. required if the backup uses volume
+ snapshot.
+ type: string
+ required:
+ - metadata
+ - volumeClaimSpec
+ type: object
+ x-kubernetes-validations:
+ - message: at least one exists for volumeSource and mountPath.
+ rule: self.volumeSource != '' || self.mountPath !=''
+ type: array
+ required:
+ - replicas
+ - templates
+ type: object
+ x-kubernetes-validations:
+ - message: forbidden to update spec.prepareDataConfig.volumeClaimsTemplate
+ rule: self == oldSelf
+ required:
+ - volumeClaimManagementPolicy
+ type: object
+ readyConfig:
+ description: configuration for the action of "postReady" phase.
+ properties:
+ connectCredential:
+ description: credential template used for creating a connection
+ credential
+ properties:
+ hostKey:
+ default: host
+ description: hostKey the map key of the host in the connection
+ credential secret
+ type: string
+ passwordKey:
+ default: password
+ description: passwordKey the map key of the password in the
+ connection credential secret
+ type: string
+ portKey:
+ default: port
+ description: portKey the map key of the port in the connection
+ credential secret
+ type: string
+ secretName:
+ description: the secret name
+ pattern: ^[a-z0-9]([a-z0-9\.\-]*[a-z0-9])?$
+ type: string
+ usernameKey:
+ default: username
+ description: usernameKey the map key of the user in the connection
+ credential secret
+ type: string
+ required:
+ - passwordKey
+ - secretName
+ - usernameKey
+ type: object
+ execAction:
+ description: configuration for exec action.
+ properties:
+ target:
+ description: execActionTarget defines the pods that need to
+ be executed for the exec action. will execute on all pods
+ that meet the conditions.
+ properties:
+ podSelector:
+ description: kubectl exec in all selected pods.
+ properties:
+ matchExpressions:
+ description: matchExpressions is a list of label selector
+ requirements. The requirements are ANDed.
+ items:
+ description: A label selector requirement is a selector
+ that contains values, a key, and an operator that
+ relates the key and values.
+ properties:
+ key:
+ description: key is the label key that the selector
+ applies to.
+ type: string
+ operator:
+ description: operator represents a key's relationship
+ to a set of values. Valid operators are In,
+ NotIn, Exists and DoesNotExist.
+ type: string
+ values:
+ description: values is an array of string values.
+ If the operator is In or NotIn, the values
+ array must be non-empty. If the operator is
+ Exists or DoesNotExist, the values array must
+ be empty. This array is replaced during a
+ strategic merge patch.
+ items:
+ type: string
+ type: array
+ required:
+ - key
+ - operator
+ type: object
+ type: array
+ matchLabels:
+ additionalProperties:
+ type: string
+ description: matchLabels is a map of {key,value} pairs.
+ A single {key,value} in the matchLabels map is equivalent
+ to an element of matchExpressions, whose key field
+ is "key", the operator is "In", and the values array
+ contains only "value". The requirements are ANDed.
+ type: object
+ type: object
+ x-kubernetes-map-type: atomic
+ required:
+ - podSelector
+ type: object
+ type: object
+ jobAction:
+ description: configuration for job action.
+ properties:
+ target:
+ description: jobActionTarget defines the pod that need to
+ be executed for the job action. will select a pod that meets
+ the conditions to execute.
+ properties:
+ podSelector:
+ description: select one of the pods which selected by
+ labels to build the job spec, such as mount required
+ volumes and inject built-in env of the selected pod.
+ properties:
+ matchExpressions:
+ description: matchExpressions is a list of label selector
+ requirements. The requirements are ANDed.
+ items:
+ description: A label selector requirement is a selector
+ that contains values, a key, and an operator that
+ relates the key and values.
+ properties:
+ key:
+ description: key is the label key that the selector
+ applies to.
+ type: string
+ operator:
+ description: operator represents a key's relationship
+ to a set of values. Valid operators are In,
+ NotIn, Exists and DoesNotExist.
+ type: string
+ values:
+ description: values is an array of string values.
+ If the operator is In or NotIn, the values
+ array must be non-empty. If the operator is
+ Exists or DoesNotExist, the values array must
+ be empty. This array is replaced during a
+ strategic merge patch.
+ items:
+ type: string
+ type: array
+ required:
+ - key
+ - operator
+ type: object
+ type: array
+ matchLabels:
+ additionalProperties:
+ type: string
+ description: matchLabels is a map of {key,value} pairs.
+ A single {key,value} in the matchLabels map is equivalent
+ to an element of matchExpressions, whose key field
+ is "key", the operator is "In", and the values array
+ contains only "value". The requirements are ANDed.
+ type: object
+ type: object
+ x-kubernetes-map-type: atomic
+ volumeMounts:
+ description: volumeMounts defines which volumes of the
+ selected pod need to be mounted on the restoring pod.
+ items:
+ description: VolumeMount describes a mounting of a Volume
+ within a container.
+ properties:
+ mountPath:
+ description: Path within the container at which
+ the volume should be mounted. Must not contain
+ ':'.
+ type: string
+ mountPropagation:
+ description: mountPropagation determines how mounts
+ are propagated from the host to container and
+ the other way around. When not set, MountPropagationNone
+ is used. This field is beta in 1.10.
+ type: string
+ name:
+ description: This must match the Name of a Volume.
+ type: string
+ readOnly:
+ description: Mounted read-only if true, read-write
+ otherwise (false or unspecified). Defaults to
+ false.
+ type: boolean
+ subPath:
+ description: Path within the volume from which the
+ container's volume should be mounted. Defaults
+ to "" (volume's root).
+ type: string
+ subPathExpr:
+ description: Expanded path within the volume from
+ which the container's volume should be mounted.
+ Behaves similarly to SubPath but environment variable
+ references $(VAR_NAME) are expanded using the
+ container's environment. Defaults to "" (volume's
+ root). SubPathExpr and SubPath are mutually exclusive.
+ type: string
+ required:
+ - mountPath
+ - name
+ type: object
+ type: array
+ required:
+ - podSelector
+ type: object
+ required:
+ - target
+ type: object
+ readinessProbe:
+ description: periodic probe of the service readiness. controller
+ will perform postReadyHooks of BackupScript.spec.restore after
+ the service readiness when readinessProbe is configured.
+ properties:
+ exec:
+ description: exec specifies the action to take.
+ properties:
+ command:
+ description: refer to container command.
+ items:
+ type: string
+ type: array
+ image:
+ description: refer to container image.
+ type: string
+ required:
+ - command
+ - image
+ type: object
+ initialDelaySeconds:
+ description: number of seconds after the container has started
+ before probe is initiated.
+ minimum: 0
+ type: integer
+ periodSeconds:
+ default: 5
+ description: how often (in seconds) to perform the probe.
+ defaults to 5 second, minimum value is 1.
+ minimum: 1
+ type: integer
+ timeoutSeconds:
+ default: 30
+ description: number of seconds after which the probe times
+ out. defaults to 30 second, minimum value is 1.
+ minimum: 1
+ type: integer
+ required:
+ - exec
+ type: object
+ type: object
+ x-kubernetes-validations:
+ - message: at least one exists for jobAction and execAction.
+ rule: has(self.jobAction) || has(self.execAction)
+ resources:
+ description: restore the specified resources of kubernetes.
+ properties:
+ included:
+ description: will restore the specified resources
+ items:
+ properties:
+ groupResource:
+ type: string
+ labelSelector:
+ description: select the specified resource for recovery
+ by label.
+ properties:
+ matchExpressions:
+ description: matchExpressions is a list of label selector
+ requirements. The requirements are ANDed.
+ items:
+ description: A label selector requirement is a selector
+ that contains values, a key, and an operator that
+ relates the key and values.
+ properties:
+ key:
+ description: key is the label key that the selector
+ applies to.
+ type: string
+ operator:
+ description: operator represents a key's relationship
+ to a set of values. Valid operators are In,
+ NotIn, Exists and DoesNotExist.
+ type: string
+ values:
+ description: values is an array of string values.
+ If the operator is In or NotIn, the values array
+ must be non-empty. If the operator is Exists
+ or DoesNotExist, the values array must be empty.
+ This array is replaced during a strategic merge
+ patch.
+ items:
+ type: string
+ type: array
+ required:
+ - key
+ - operator
+ type: object
+ type: array
+ matchLabels:
+ additionalProperties:
+ type: string
+ description: matchLabels is a map of {key,value} pairs.
+ A single {key,value} in the matchLabels map is equivalent
+ to an element of matchExpressions, whose key field
+ is "key", the operator is "In", and the values array
+ contains only "value". The requirements are ANDed.
+ type: object
+ type: object
+ x-kubernetes-map-type: atomic
+ required:
+ - groupResource
+ type: object
+ type: array
+ type: object
+ x-kubernetes-validations:
+ - message: forbidden to update spec.resources
+ rule: self == oldSelf
+ restoreTime:
+ description: restore according to a specified point in time.
+ pattern: ^\d{4}-\d{2}-\d{2}T\d{2}:\d{2}:\d{2}Z$
+ type: string
+ x-kubernetes-validations:
+ - message: forbidden to update spec.restoreTime
+ rule: self == oldSelf
+ serviceAccountName:
+ description: service account name which needs for recovery pod.
+ type: string
+ required:
+ - backup
+ type: object
+ status:
+ description: RestoreStatus defines the observed state of Restore
+ properties:
+ actions:
+ description: recorded all restore actions performed.
+ properties:
+ postReady:
+ description: record the actions for postReady phase.
+ items:
+ properties:
+ backupName:
+ description: which backup's restore action belongs to.
+ type: string
+ endTime:
+ description: endTime is the completion time for the restore
+ job.
+ format: date-time
+ type: string
+ message:
+ description: message is a human readable message indicating
+ details about the object condition.
+ type: string
+ name:
+ description: name describes the name of the recovery action
+ based on the current backup.
+ type: string
+ objectKey:
+ description: the execution object of the restore action.
+ type: string
+ startTime:
+ description: startTime is the start time for the restore
+ job.
+ format: date-time
+ type: string
+ status:
+ description: the status of this action.
+ enum:
+ - Processing
+ - Completed
+ - Failed
+ type: string
+ required:
+ - backupName
+ - name
+ - objectKey
+ type: object
+ type: array
+ prepareData:
+ description: record the actions for prepareData phase.
+ items:
+ properties:
+ backupName:
+ description: which backup's restore action belongs to.
+ type: string
+ endTime:
+ description: endTime is the completion time for the restore
+ job.
+ format: date-time
+ type: string
+ message:
+ description: message is a human readable message indicating
+ details about the object condition.
+ type: string
+ name:
+ description: name describes the name of the recovery action
+ based on the current backup.
+ type: string
+ objectKey:
+ description: the execution object of the restore action.
+ type: string
+ startTime:
+ description: startTime is the start time for the restore
+ job.
+ format: date-time
+ type: string
+ status:
+ description: the status of this action.
+ enum:
+ - Processing
+ - Completed
+ - Failed
+ type: string
+ required:
+ - backupName
+ - name
+ - objectKey
+ type: object
+ type: array
+ type: object
+ completionTimestamp:
+ description: Date/time when the restore finished being processed.
+ format: date-time
+ type: string
+ conditions:
+ description: describe current state of restore API Resource, like
+ warning.
+ items:
+ description: "Condition contains details for one aspect of the current
+ state of this API Resource. --- This struct is intended for direct
+ use as an array at the field path .status.conditions. For example,
+ \n type FooStatus struct{ // Represents the observations of a
+ foo's current state. // Known .status.conditions.type are: \"Available\",
+ \"Progressing\", and \"Degraded\" // +patchMergeKey=type // +patchStrategy=merge
+ // +listType=map // +listMapKey=type Conditions []metav1.Condition
+ `json:\"conditions,omitempty\" patchStrategy:\"merge\" patchMergeKey:\"type\"
+ protobuf:\"bytes,1,rep,name=conditions\"` \n // other fields }"
+ properties:
+ lastTransitionTime:
+ description: lastTransitionTime is the last time the condition
+ transitioned from one status to another. This should be when
+ the underlying condition changed. If that is not known, then
+ using the time when the API field changed is acceptable.
+ format: date-time
+ type: string
+ message:
+ description: message is a human readable message indicating
+ details about the transition. This may be an empty string.
+ maxLength: 32768
+ type: string
+ observedGeneration:
+ description: observedGeneration represents the .metadata.generation
+ that the condition was set based upon. For instance, if .metadata.generation
+ is currently 12, but the .status.conditions[x].observedGeneration
+ is 9, the condition is out of date with respect to the current
+ state of the instance.
+ format: int64
+ minimum: 0
+ type: integer
+ reason:
+ description: reason contains a programmatic identifier indicating
+ the reason for the condition's last transition. Producers
+ of specific condition types may define expected values and
+ meanings for this field, and whether the values are considered
+ a guaranteed API. The value should be a CamelCase string.
+ This field may not be empty.
+ maxLength: 1024
+ minLength: 1
+ pattern: ^[A-Za-z]([A-Za-z0-9_,:]*[A-Za-z0-9_])?$
+ type: string
+ status:
+ description: status of the condition, one of True, False, Unknown.
+ enum:
+ - "True"
+ - "False"
+ - Unknown
+ type: string
+ type:
+ description: type of condition in CamelCase or in foo.example.com/CamelCase.
+ --- Many .condition.type values are consistent across resources
+ like Available, but because arbitrary conditions can be useful
+ (see .node.status.conditions), the ability to deconflict is
+ important. The regex it matches is (dns1123SubdomainFmt/)?(qualifiedNameFmt)
+ maxLength: 316
+ pattern: ^([a-z0-9]([-a-z0-9]*[a-z0-9])?(\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*/)?(([A-Za-z0-9][-A-Za-z0-9_.]*)?[A-Za-z0-9])$
+ type: string
+ required:
+ - lastTransitionTime
+ - message
+ - reason
+ - status
+ - type
+ type: object
+ type: array
+ duration:
+ description: The duration time of restore execution. When converted
+ to a string, the form is "1h2m0.5s".
+ type: string
+ phase:
+ description: RestorePhase The current phase. Valid values are Running,
+ Completed, Failed, Deleting.
+ enum:
+ - Running
+ - Completed
+ - Failed
+ - Deleting
+ type: string
+ startTimestamp:
+ description: Date/time when the restore started being processed.
+ format: date-time
+ type: string
+ type: object
+ type: object
+ served: true
+ storage: true
+ subresources:
+ status: {}
diff --git a/deploy/helm/crds/extensions.kubeblocks.io_addons.yaml b/deploy/helm/crds/extensions.kubeblocks.io_addons.yaml
index f170da7b08c..80f4c9f946b 100644
--- a/deploy/helm/crds/extensions.kubeblocks.io_addons.yaml
+++ b/deploy/helm/crds/extensions.kubeblocks.io_addons.yaml
@@ -2,8 +2,7 @@ apiVersion: apiextensions.k8s.io/v1
kind: CustomResourceDefinition
metadata:
annotations:
- controller-gen.kubebuilder.io/version: v0.9.0
- creationTimestamp: null
+ controller-gen.kubebuilder.io/version: v0.12.1
labels:
app.kubernetes.io/name: kubeblocks
name: addons.extensions.kubeblocks.io
@@ -172,12 +171,15 @@ spec:
properties:
key:
description: The selector key. Valid values are KubeVersion,
- KubeGitVersion. "KubeVersion" the semver expression
- of Kubernetes versions, i.e., v1.24. "KubeGitVersion"
- may contain distro. info., i.e., v1.24.4+eks.
+ KubeGitVersion and KubeProvider. "KubeVersion" the semver
+ expression of Kubernetes versions, i.e., v1.24. "KubeGitVersion"
+ may contain distro. info., i.e., v1.24.4+eks. "KubeProvider"
+ the Kubernetes provider, i.e., aws,gcp,azure,huaweiCloud,tencentCloud
+ etc.
enum:
- KubeGitVersion
- KubeVersion
+ - KubeProvider
type: string
operator:
description: "Represents a key's relationship to a set
@@ -562,12 +564,15 @@ spec:
properties:
key:
description: The selector key. Valid values are KubeVersion,
- KubeGitVersion. "KubeVersion" the semver expression of
- Kubernetes versions, i.e., v1.24. "KubeGitVersion" may
- contain distro. info., i.e., v1.24.4+eks.
+ KubeGitVersion and KubeProvider. "KubeVersion" the semver
+ expression of Kubernetes versions, i.e., v1.24. "KubeGitVersion"
+ may contain distro. info., i.e., v1.24.4+eks. "KubeProvider"
+ the Kubernetes provider, i.e., aws,gcp,azure,huaweiCloud,tencentCloud
+ etc.
enum:
- KubeGitVersion
- KubeVersion
+ - KubeProvider
type: string
operator:
description: "Represents a key's relationship to a set of
diff --git a/deploy/helm/crds/storage.kubeblocks.io_storageproviders.yaml b/deploy/helm/crds/storage.kubeblocks.io_storageproviders.yaml
index af55709f62e..5c2b75b65b3 100644
--- a/deploy/helm/crds/storage.kubeblocks.io_storageproviders.yaml
+++ b/deploy/helm/crds/storage.kubeblocks.io_storageproviders.yaml
@@ -2,8 +2,7 @@ apiVersion: apiextensions.k8s.io/v1
kind: CustomResourceDefinition
metadata:
annotations:
- controller-gen.kubebuilder.io/version: v0.9.0
- creationTimestamp: null
+ controller-gen.kubebuilder.io/version: v0.12.1
labels:
app.kubernetes.io/name: kubeblocks
name: storageproviders.storage.kubeblocks.io
@@ -18,7 +17,17 @@ spec:
singular: storageprovider
scope: Cluster
versions:
- - name: v1alpha1
+ - additionalPrinterColumns:
+ - jsonPath: .status.phase
+ name: STATUS
+ type: string
+ - jsonPath: .spec.csiDriverName
+ name: CSIDRIVER
+ type: string
+ - jsonPath: .metadata.creationTimestamp
+ name: AGE
+ type: date
+ name: v1alpha1
schema:
openAPIV3Schema:
description: StorageProvider is the Schema for the storageproviders API StorageProvider
@@ -48,6 +57,11 @@ spec:
by the CSI driver. The template will be rendered with the following
variables: - Parameters: a map of parameters defined in the ParametersSchema.'
type: string
+ datasafedConfigTemplate:
+ description: 'A Go template for rendering a config used by the datasafed
+ command. The template will be rendered with the following variables:
+ - Parameters: a map of parameters defined in the ParametersSchema.'
+ type: string
parametersSchema:
description: The schema describes the parameters required by this
StorageProvider, when rendering the templates.
@@ -64,6 +78,12 @@ spec:
type: object
x-kubernetes-preserve-unknown-fields: true
type: object
+ persistentVolumeClaimTemplate:
+ description: 'A Go template for rendering a PersistentVolumeClaim.
+ The template will be rendered with the following variables: - Parameters:
+ a map of parameters defined in the ParametersSchema. - GeneratedStorageClassName:
+ the name of the storage class generated with the StorageClassTemplate.'
+ type: string
storageClassTemplate:
description: 'A Go template for rendering a storage class which will
be used by the CSI driver. The template will be rendered with the
diff --git a/deploy/helm/crds/workloads.kubeblocks.io_replicatedstatemachines.yaml b/deploy/helm/crds/workloads.kubeblocks.io_replicatedstatemachines.yaml
index 2836ac341b7..6e84668baf6 100644
--- a/deploy/helm/crds/workloads.kubeblocks.io_replicatedstatemachines.yaml
+++ b/deploy/helm/crds/workloads.kubeblocks.io_replicatedstatemachines.yaml
@@ -2,8 +2,7 @@ apiVersion: apiextensions.k8s.io/v1
kind: CustomResourceDefinition
metadata:
annotations:
- controller-gen.kubebuilder.io/version: v0.9.0
- creationTimestamp: null
+ controller-gen.kubebuilder.io/version: v0.12.1
labels:
app.kubernetes.io/name: kubeblocks
name: replicatedstatemachines.workloads.kubeblocks.io
@@ -41,7 +40,7 @@ spec:
schema:
openAPIV3Schema:
description: ReplicatedStateMachine is the Schema for the replicatedstatemachines
- API
+ API.
properties:
apiVersion:
description: 'APIVersion defines the versioned schema of this representation
@@ -287,11 +286,10 @@ spec:
supports specifying the loadBalancerIP when a load balancer
is created. This field will be ignored if the cloud-provider
does not support the feature. Deprecated: This field was
- under-specified and its meaning varies across implementations,
- and it cannot support dual-stack. As of Kubernetes v1.24,
- users are encouraged to use implementation-specific annotations
- when available. This field may be removed in a future
- API version.'
+ under-specified and its meaning varies across implementations.
+ Using it is non-portable and it may not support dual-stack.
+ Users are encouraged to use implementation-specific annotations
+ when available.'
type: string
loadBalancerSourceRanges:
description: 'If specified and supported by the platform,
@@ -310,12 +308,21 @@ spec:
port.
properties:
appProtocol:
- description: The application protocol for this port.
+ description: "The application protocol for this port.
+ This is used as a hint for implementations to offer
+ richer behavior for protocols that they understand.
This field follows standard Kubernetes label syntax.
- Un-prefixed names are reserved for IANA standard
- service names (as per RFC-6335 and https://www.iana.org/assignments/service-names).
- Non-standard protocols should use prefixed names
- such as mycompany.com/my-custom-protocol.
+ Valid values are either: \n * Un-prefixed protocol
+ names - reserved for IANA standard service names
+ (as per RFC-6335 and https://www.iana.org/assignments/service-names).
+ \n * Kubernetes-defined prefixed names: * 'kubernetes.io/h2c'
+ - HTTP/2 over cleartext as described in https://www.rfc-editor.org/rfc/rfc7540
+ * 'kubernetes.io/ws' - WebSocket over cleartext
+ as described in https://www.rfc-editor.org/rfc/rfc6455
+ * 'kubernetes.io/wss' - WebSocket over TLS as described
+ in https://www.rfc-editor.org/rfc/rfc6455 \n * Other
+ protocols should use implementation-defined prefixed
+ names such as mycompany.com/my-custom-protocol."
type: string
name:
description: The name of this port within the service.
@@ -630,6 +637,7 @@ spec:
required:
- key
type: object
+ x-kubernetes-map-type: atomic
fieldRef:
description: 'Selects a field of the pod: supports metadata.name,
metadata.namespace, `metadata.labels['''']`, `metadata.annotations['''']`,
@@ -647,6 +655,7 @@ spec:
required:
- fieldPath
type: object
+ x-kubernetes-map-type: atomic
resourceFieldRef:
description: 'Selects a resource of the container: only
resources limits and requests (limits.cpu, limits.memory,
@@ -671,6 +680,7 @@ spec:
required:
- resource
type: object
+ x-kubernetes-map-type: atomic
secretKeyRef:
description: Selects a key of a secret in the pod's namespace
properties:
@@ -690,6 +700,7 @@ spec:
required:
- key
type: object
+ x-kubernetes-map-type: atomic
type: object
type: object
username:
@@ -728,6 +739,7 @@ spec:
required:
- key
type: object
+ x-kubernetes-map-type: atomic
fieldRef:
description: 'Selects a field of the pod: supports metadata.name,
metadata.namespace, `metadata.labels['''']`, `metadata.annotations['''']`,
@@ -745,6 +757,7 @@ spec:
required:
- fieldPath
type: object
+ x-kubernetes-map-type: atomic
resourceFieldRef:
description: 'Selects a resource of the container: only
resources limits and requests (limits.cpu, limits.memory,
@@ -769,6 +782,7 @@ spec:
required:
- resource
type: object
+ x-kubernetes-map-type: atomic
secretKeyRef:
description: Selects a key of a secret in the pod's namespace
properties:
@@ -788,6 +802,7 @@ spec:
required:
- key
type: object
+ x-kubernetes-map-type: atomic
type: object
type: object
required:
@@ -970,6 +985,15 @@ spec:
- command
type: object
type: array
+ roleUpdateMechanism:
+ default: None
+ description: RoleUpdateMechanism specifies the way how pod role
+ label being updated.
+ enum:
+ - ReadinessProbeEventUpdate
+ - DirectAPIServerEventUpdate
+ - None
+ type: string
successThreshold:
default: 1
description: Minimum consecutive successes for the probe to be
@@ -1063,6 +1087,7 @@ spec:
are ANDed.
type: object
type: object
+ x-kubernetes-map-type: atomic
service:
description: service defines the behavior of a service spec. provides
read-write service https://git.k8s.io/community/contributors/devel/sig-architecture/api-conventions.md#spec-and-status
@@ -1284,11 +1309,10 @@ spec:
supports specifying the loadBalancerIP when a load balancer
is created. This field will be ignored if the cloud-provider
does not support the feature. Deprecated: This field was
- under-specified and its meaning varies across implementations,
- and it cannot support dual-stack. As of Kubernetes v1.24,
- users are encouraged to use implementation-specific annotations
- when available. This field may be removed in a future API
- version.'
+ under-specified and its meaning varies across implementations.
+ Using it is non-portable and it may not support dual-stack.
+ Users are encouraged to use implementation-specific annotations
+ when available.'
type: string
loadBalancerSourceRanges:
description: 'If specified and supported by the platform,
@@ -1307,12 +1331,21 @@ spec:
port.
properties:
appProtocol:
- description: The application protocol for this port.
+ description: "The application protocol for this port.
+ This is used as a hint for implementations to offer
+ richer behavior for protocols that they understand.
This field follows standard Kubernetes label syntax.
- Un-prefixed names are reserved for IANA standard service
- names (as per RFC-6335 and https://www.iana.org/assignments/service-names).
- Non-standard protocols should use prefixed names such
- as mycompany.com/my-custom-protocol.
+ Valid values are either: \n * Un-prefixed protocol
+ names - reserved for IANA standard service names (as
+ per RFC-6335 and https://www.iana.org/assignments/service-names).
+ \n * Kubernetes-defined prefixed names: * 'kubernetes.io/h2c'
+ - HTTP/2 over cleartext as described in https://www.rfc-editor.org/rfc/rfc7540
+ * 'kubernetes.io/ws' - WebSocket over cleartext as
+ described in https://www.rfc-editor.org/rfc/rfc6455
+ * 'kubernetes.io/wss' - WebSocket over TLS as described
+ in https://www.rfc-editor.org/rfc/rfc6455 \n * Other
+ protocols should use implementation-defined prefixed
+ names such as mycompany.com/my-custom-protocol."
type: string
name:
description: The name of this port within the service.
@@ -1729,6 +1762,7 @@ spec:
type: object
type: array
type: object
+ x-kubernetes-map-type: atomic
weight:
description: Weight associated with matching
the corresponding nodeSelectorTerm, in the
@@ -1835,10 +1869,12 @@ spec:
type: object
type: array
type: object
+ x-kubernetes-map-type: atomic
type: array
required:
- nodeSelectorTerms
type: object
+ x-kubernetes-map-type: atomic
type: object
podAffinity:
description: Describes pod affinity scheduling rules (e.g.
@@ -1923,6 +1959,7 @@ spec:
ANDed.
type: object
type: object
+ x-kubernetes-map-type: atomic
namespaceSelector:
description: A label query over the set
of namespaces that the term applies to.
@@ -1984,6 +2021,7 @@ spec:
ANDed.
type: object
type: object
+ x-kubernetes-map-type: atomic
namespaces:
description: namespaces specifies a static
list of namespace names that the term
@@ -2093,6 +2131,7 @@ spec:
only "value". The requirements are ANDed.
type: object
type: object
+ x-kubernetes-map-type: atomic
namespaceSelector:
description: A label query over the set of namespaces
that the term applies to. The term is applied
@@ -2150,6 +2189,7 @@ spec:
only "value". The requirements are ANDed.
type: object
type: object
+ x-kubernetes-map-type: atomic
namespaces:
description: namespaces specifies a static list
of namespace names that the term applies to.
@@ -2259,6 +2299,7 @@ spec:
ANDed.
type: object
type: object
+ x-kubernetes-map-type: atomic
namespaceSelector:
description: A label query over the set
of namespaces that the term applies to.
@@ -2320,6 +2361,7 @@ spec:
ANDed.
type: object
type: object
+ x-kubernetes-map-type: atomic
namespaces:
description: namespaces specifies a static
list of namespace names that the term
@@ -2429,6 +2471,7 @@ spec:
only "value". The requirements are ANDed.
type: object
type: object
+ x-kubernetes-map-type: atomic
namespaceSelector:
description: A label query over the set of namespaces
that the term applies to. The term is applied
@@ -2486,6 +2529,7 @@ spec:
only "value". The requirements are ANDed.
type: object
type: object
+ x-kubernetes-map-type: atomic
namespaces:
description: namespaces specifies a static list
of namespace names that the term applies to.
@@ -2602,6 +2646,7 @@ spec:
required:
- key
type: object
+ x-kubernetes-map-type: atomic
fieldRef:
description: 'Selects a field of the pod:
supports metadata.name, metadata.namespace,
@@ -2621,6 +2666,7 @@ spec:
required:
- fieldPath
type: object
+ x-kubernetes-map-type: atomic
resourceFieldRef:
description: 'Selects a resource of the container:
only resources limits and requests (limits.cpu,
@@ -2647,6 +2693,7 @@ spec:
required:
- resource
type: object
+ x-kubernetes-map-type: atomic
secretKeyRef:
description: Selects a key of a secret in
the pod's namespace
@@ -2669,6 +2716,7 @@ spec:
required:
- key
type: object
+ x-kubernetes-map-type: atomic
type: object
required:
- name
@@ -2701,6 +2749,7 @@ spec:
must be defined
type: boolean
type: object
+ x-kubernetes-map-type: atomic
prefix:
description: An optional identifier to prepend
to each key in the ConfigMap. Must be a C_IDENTIFIER.
@@ -2719,6 +2768,7 @@ spec:
be defined
type: boolean
type: object
+ x-kubernetes-map-type: atomic
type: object
type: array
image:
@@ -2781,7 +2831,10 @@ spec:
header to be used in HTTP probes
properties:
name:
- description: The header field name
+ description: The header field name.
+ This will be canonicalized upon
+ output, so case-variant names will
+ be understood as the same header.
type: string
value:
description: The header field value
@@ -2885,7 +2938,10 @@ spec:
header to be used in HTTP probes
properties:
name:
- description: The header field name
+ description: The header field name.
+ This will be canonicalized upon
+ output, so case-variant names will
+ be understood as the same header.
type: string
value:
description: The header field value
@@ -2970,8 +3026,7 @@ spec:
type: integer
grpc:
description: GRPC specifies an action involving
- a GRPC port. This is a beta field and requires
- enabling GRPCContainerProbe feature gate.
+ a GRPC port.
properties:
port:
description: Port number of the gRPC service.
@@ -3005,7 +3060,10 @@ spec:
header to be used in HTTP probes
properties:
name:
- description: The header field name
+ description: The header field name. This
+ will be canonicalized upon output, so
+ case-variant names will be understood
+ as the same header.
type: string
value:
description: The header field value
@@ -3182,8 +3240,7 @@ spec:
type: integer
grpc:
description: GRPC specifies an action involving
- a GRPC port. This is a beta field and requires
- enabling GRPCContainerProbe feature gate.
+ a GRPC port.
properties:
port:
description: Port number of the gRPC service.
@@ -3217,7 +3274,10 @@ spec:
header to be used in HTTP probes
properties:
name:
- description: The header field name
+ description: The header field name. This
+ will be canonicalized upon output, so
+ case-variant names will be understood
+ as the same header.
type: string
value:
description: The header field value
@@ -3309,6 +3369,28 @@ spec:
format: int32
type: integer
type: object
+ resizePolicy:
+ description: Resources resize policy for the container.
+ items:
+ description: ContainerResizePolicy represents resource
+ resize policy for the container.
+ properties:
+ resourceName:
+ description: 'Name of the resource to which this
+ resource resize policy applies. Supported values:
+ cpu, memory.'
+ type: string
+ restartPolicy:
+ description: Restart policy to apply when specified
+ resource is resized. If not specified, it defaults
+ to NotRequired.
+ type: string
+ required:
+ - resourceName
+ - restartPolicy
+ type: object
+ type: array
+ x-kubernetes-list-type: atomic
resources:
description: 'Compute Resources required by this container.
Cannot be updated. More info: https://kubernetes.io/docs/concepts/configuration/manage-resources-containers/'
@@ -3358,9 +3440,32 @@ spec:
of compute resources required. If Requests is
omitted for a container, it defaults to Limits
if that is explicitly specified, otherwise to
- an implementation-defined value. More info: https://kubernetes.io/docs/concepts/configuration/manage-resources-containers/'
+ an implementation-defined value. Requests cannot
+ exceed Limits. More info: https://kubernetes.io/docs/concepts/configuration/manage-resources-containers/'
type: object
type: object
+ restartPolicy:
+ description: 'RestartPolicy defines the restart behavior
+ of individual containers in a pod. This field may
+ only be set for init containers, and the only allowed
+ value is "Always". For non-init containers or when
+ this field is not specified, the restart behavior
+ is defined by the Pod''s restart policy and the container
+ type. Setting the RestartPolicy as "Always" for the
+ init container will have the following effect: this
+ init container will be continually restarted on exit
+ until all regular containers have terminated. Once
+ all regular containers have completed, all init containers
+ with restartPolicy "Always" will be shut down. This
+ lifecycle differs from normal init containers and
+ is often referred to as a "sidecar" container. Although
+ this init container still starts in the init container
+ sequence, it does not wait for the container to complete
+ before proceeding to the next init container. Instead,
+ the next init container starts immediately after this
+ init container is started, or after any startupProbe
+ has successfully completed.'
+ type: string
securityContext:
description: 'SecurityContext defines the security options
the container should be run with. If set, the fields
@@ -3490,7 +3595,8 @@ spec:
The profile must be preconfigured on the node
to work. Must be a descending path, relative
to the kubelet's configured seccomp profile
- location. Must only be set if type is "Localhost".
+ location. Must be set if type is "Localhost".
+ Must NOT be set for any other type.
type: string
type:
description: "type indicates which kind of seccomp
@@ -3526,14 +3632,10 @@ spec:
hostProcess:
description: HostProcess determines if a container
should be run as a 'Host Process' container.
- This field is alpha-level and will only be
- honored by components that enable the WindowsHostProcessContainers
- feature flag. Setting this field without the
- feature flag will result in errors when validating
- the Pod. All of a Pod's containers must have
- the same effective HostProcess value (it is
- not allowed to have a mix of HostProcess containers
- and non-HostProcess containers). In addition,
+ All of a Pod's containers must have the same
+ effective HostProcess value (it is not allowed
+ to have a mix of HostProcess containers and
+ non-HostProcess containers). In addition,
if HostProcess is true then HostNetwork must
also be set to true.
type: boolean
@@ -3584,8 +3686,7 @@ spec:
type: integer
grpc:
description: GRPC specifies an action involving
- a GRPC port. This is a beta field and requires
- enabling GRPCContainerProbe feature gate.
+ a GRPC port.
properties:
port:
description: Port number of the gRPC service.
@@ -3619,7 +3720,10 @@ spec:
header to be used in HTTP probes
properties:
name:
- description: The header field name
+ description: The header field name. This
+ will be canonicalized upon output, so
+ case-variant names will be understood
+ as the same header.
type: string
value:
description: The header field value
@@ -3981,6 +4085,7 @@ spec:
required:
- key
type: object
+ x-kubernetes-map-type: atomic
fieldRef:
description: 'Selects a field of the pod:
supports metadata.name, metadata.namespace,
@@ -4000,6 +4105,7 @@ spec:
required:
- fieldPath
type: object
+ x-kubernetes-map-type: atomic
resourceFieldRef:
description: 'Selects a resource of the container:
only resources limits and requests (limits.cpu,
@@ -4026,6 +4132,7 @@ spec:
required:
- resource
type: object
+ x-kubernetes-map-type: atomic
secretKeyRef:
description: Selects a key of a secret in
the pod's namespace
@@ -4048,6 +4155,7 @@ spec:
required:
- key
type: object
+ x-kubernetes-map-type: atomic
type: object
required:
- name
@@ -4080,6 +4188,7 @@ spec:
must be defined
type: boolean
type: object
+ x-kubernetes-map-type: atomic
prefix:
description: An optional identifier to prepend
to each key in the ConfigMap. Must be a C_IDENTIFIER.
@@ -4098,6 +4207,7 @@ spec:
be defined
type: boolean
type: object
+ x-kubernetes-map-type: atomic
type: object
type: array
image:
@@ -4156,7 +4266,10 @@ spec:
header to be used in HTTP probes
properties:
name:
- description: The header field name
+ description: The header field name.
+ This will be canonicalized upon
+ output, so case-variant names will
+ be understood as the same header.
type: string
value:
description: The header field value
@@ -4260,7 +4373,10 @@ spec:
header to be used in HTTP probes
properties:
name:
- description: The header field name
+ description: The header field name.
+ This will be canonicalized upon
+ output, so case-variant names will
+ be understood as the same header.
type: string
value:
description: The header field value
@@ -4343,8 +4459,7 @@ spec:
type: integer
grpc:
description: GRPC specifies an action involving
- a GRPC port. This is a beta field and requires
- enabling GRPCContainerProbe feature gate.
+ a GRPC port.
properties:
port:
description: Port number of the gRPC service.
@@ -4378,7 +4493,10 @@ spec:
header to be used in HTTP probes
properties:
name:
- description: The header field name
+ description: The header field name. This
+ will be canonicalized upon output, so
+ case-variant names will be understood
+ as the same header.
type: string
value:
description: The header field value
@@ -4546,8 +4664,7 @@ spec:
type: integer
grpc:
description: GRPC specifies an action involving
- a GRPC port. This is a beta field and requires
- enabling GRPCContainerProbe feature gate.
+ a GRPC port.
properties:
port:
description: Port number of the gRPC service.
@@ -4581,7 +4698,10 @@ spec:
header to be used in HTTP probes
properties:
name:
- description: The header field name
+ description: The header field name. This
+ will be canonicalized upon output, so
+ case-variant names will be understood
+ as the same header.
type: string
value:
description: The header field value
@@ -4673,6 +4793,28 @@ spec:
format: int32
type: integer
type: object
+ resizePolicy:
+ description: Resources resize policy for the container.
+ items:
+ description: ContainerResizePolicy represents resource
+ resize policy for the container.
+ properties:
+ resourceName:
+ description: 'Name of the resource to which this
+ resource resize policy applies. Supported values:
+ cpu, memory.'
+ type: string
+ restartPolicy:
+ description: Restart policy to apply when specified
+ resource is resized. If not specified, it defaults
+ to NotRequired.
+ type: string
+ required:
+ - resourceName
+ - restartPolicy
+ type: object
+ type: array
+ x-kubernetes-list-type: atomic
resources:
description: Resources are not allowed for ephemeral
containers. Ephemeral containers use spare resources
@@ -4723,9 +4865,16 @@ spec:
of compute resources required. If Requests is
omitted for a container, it defaults to Limits
if that is explicitly specified, otherwise to
- an implementation-defined value. More info: https://kubernetes.io/docs/concepts/configuration/manage-resources-containers/'
+ an implementation-defined value. Requests cannot
+ exceed Limits. More info: https://kubernetes.io/docs/concepts/configuration/manage-resources-containers/'
type: object
type: object
+ restartPolicy:
+ description: Restart policy for the container to manage
+ the restart behavior of each container within a pod.
+ This may only be set for init containers. You cannot
+ set this field on ephemeral containers.
+ type: string
securityContext:
description: 'Optional: SecurityContext defines the
security options the ephemeral container should be
@@ -4855,7 +5004,8 @@ spec:
The profile must be preconfigured on the node
to work. Must be a descending path, relative
to the kubelet's configured seccomp profile
- location. Must only be set if type is "Localhost".
+ location. Must be set if type is "Localhost".
+ Must NOT be set for any other type.
type: string
type:
description: "type indicates which kind of seccomp
@@ -4891,14 +5041,10 @@ spec:
hostProcess:
description: HostProcess determines if a container
should be run as a 'Host Process' container.
- This field is alpha-level and will only be
- honored by components that enable the WindowsHostProcessContainers
- feature flag. Setting this field without the
- feature flag will result in errors when validating
- the Pod. All of a Pod's containers must have
- the same effective HostProcess value (it is
- not allowed to have a mix of HostProcess containers
- and non-HostProcess containers). In addition,
+ All of a Pod's containers must have the same
+ effective HostProcess value (it is not allowed
+ to have a mix of HostProcess containers and
+ non-HostProcess containers). In addition,
if HostProcess is true then HostNetwork must
also be set to true.
type: boolean
@@ -4941,8 +5087,7 @@ spec:
type: integer
grpc:
description: GRPC specifies an action involving
- a GRPC port. This is a beta field and requires
- enabling GRPCContainerProbe feature gate.
+ a GRPC port.
properties:
port:
description: Port number of the gRPC service.
@@ -4976,7 +5121,10 @@ spec:
header to be used in HTTP probes
properties:
name:
- description: The header field name
+ description: The header field name. This
+ will be canonicalized upon output, so
+ case-variant names will be understood
+ as the same header.
type: string
value:
description: The header field value
@@ -5267,6 +5415,7 @@ spec:
TODO: Add other useful fields. apiVersion, kind, uid?'
type: string
type: object
+ x-kubernetes-map-type: atomic
type: array
initContainers:
description: 'List of initialization containers belonging
@@ -5364,6 +5513,7 @@ spec:
required:
- key
type: object
+ x-kubernetes-map-type: atomic
fieldRef:
description: 'Selects a field of the pod:
supports metadata.name, metadata.namespace,
@@ -5383,6 +5533,7 @@ spec:
required:
- fieldPath
type: object
+ x-kubernetes-map-type: atomic
resourceFieldRef:
description: 'Selects a resource of the container:
only resources limits and requests (limits.cpu,
@@ -5409,6 +5560,7 @@ spec:
required:
- resource
type: object
+ x-kubernetes-map-type: atomic
secretKeyRef:
description: Selects a key of a secret in
the pod's namespace
@@ -5431,6 +5583,7 @@ spec:
required:
- key
type: object
+ x-kubernetes-map-type: atomic
type: object
required:
- name
@@ -5463,6 +5616,7 @@ spec:
must be defined
type: boolean
type: object
+ x-kubernetes-map-type: atomic
prefix:
description: An optional identifier to prepend
to each key in the ConfigMap. Must be a C_IDENTIFIER.
@@ -5481,6 +5635,7 @@ spec:
be defined
type: boolean
type: object
+ x-kubernetes-map-type: atomic
type: object
type: array
image:
@@ -5543,7 +5698,10 @@ spec:
header to be used in HTTP probes
properties:
name:
- description: The header field name
+ description: The header field name.
+ This will be canonicalized upon
+ output, so case-variant names will
+ be understood as the same header.
type: string
value:
description: The header field value
@@ -5647,7 +5805,10 @@ spec:
header to be used in HTTP probes
properties:
name:
- description: The header field name
+ description: The header field name.
+ This will be canonicalized upon
+ output, so case-variant names will
+ be understood as the same header.
type: string
value:
description: The header field value
@@ -5732,8 +5893,7 @@ spec:
type: integer
grpc:
description: GRPC specifies an action involving
- a GRPC port. This is a beta field and requires
- enabling GRPCContainerProbe feature gate.
+ a GRPC port.
properties:
port:
description: Port number of the gRPC service.
@@ -5767,7 +5927,10 @@ spec:
header to be used in HTTP probes
properties:
name:
- description: The header field name
+ description: The header field name. This
+ will be canonicalized upon output, so
+ case-variant names will be understood
+ as the same header.
type: string
value:
description: The header field value
@@ -5944,8 +6107,7 @@ spec:
type: integer
grpc:
description: GRPC specifies an action involving
- a GRPC port. This is a beta field and requires
- enabling GRPCContainerProbe feature gate.
+ a GRPC port.
properties:
port:
description: Port number of the gRPC service.
@@ -5979,7 +6141,10 @@ spec:
header to be used in HTTP probes
properties:
name:
- description: The header field name
+ description: The header field name. This
+ will be canonicalized upon output, so
+ case-variant names will be understood
+ as the same header.
type: string
value:
description: The header field value
@@ -6071,6 +6236,28 @@ spec:
format: int32
type: integer
type: object
+ resizePolicy:
+ description: Resources resize policy for the container.
+ items:
+ description: ContainerResizePolicy represents resource
+ resize policy for the container.
+ properties:
+ resourceName:
+ description: 'Name of the resource to which this
+ resource resize policy applies. Supported values:
+ cpu, memory.'
+ type: string
+ restartPolicy:
+ description: Restart policy to apply when specified
+ resource is resized. If not specified, it defaults
+ to NotRequired.
+ type: string
+ required:
+ - resourceName
+ - restartPolicy
+ type: object
+ type: array
+ x-kubernetes-list-type: atomic
resources:
description: 'Compute Resources required by this container.
Cannot be updated. More info: https://kubernetes.io/docs/concepts/configuration/manage-resources-containers/'
@@ -6120,9 +6307,32 @@ spec:
of compute resources required. If Requests is
omitted for a container, it defaults to Limits
if that is explicitly specified, otherwise to
- an implementation-defined value. More info: https://kubernetes.io/docs/concepts/configuration/manage-resources-containers/'
+ an implementation-defined value. Requests cannot
+ exceed Limits. More info: https://kubernetes.io/docs/concepts/configuration/manage-resources-containers/'
type: object
type: object
+ restartPolicy:
+ description: 'RestartPolicy defines the restart behavior
+ of individual containers in a pod. This field may
+ only be set for init containers, and the only allowed
+ value is "Always". For non-init containers or when
+ this field is not specified, the restart behavior
+ is defined by the Pod''s restart policy and the container
+ type. Setting the RestartPolicy as "Always" for the
+ init container will have the following effect: this
+ init container will be continually restarted on exit
+ until all regular containers have terminated. Once
+ all regular containers have completed, all init containers
+ with restartPolicy "Always" will be shut down. This
+ lifecycle differs from normal init containers and
+ is often referred to as a "sidecar" container. Although
+ this init container still starts in the init container
+ sequence, it does not wait for the container to complete
+ before proceeding to the next init container. Instead,
+ the next init container starts immediately after this
+ init container is started, or after any startupProbe
+ has successfully completed.'
+ type: string
securityContext:
description: 'SecurityContext defines the security options
the container should be run with. If set, the fields
@@ -6252,7 +6462,8 @@ spec:
The profile must be preconfigured on the node
to work. Must be a descending path, relative
to the kubelet's configured seccomp profile
- location. Must only be set if type is "Localhost".
+ location. Must be set if type is "Localhost".
+ Must NOT be set for any other type.
type: string
type:
description: "type indicates which kind of seccomp
@@ -6288,14 +6499,10 @@ spec:
hostProcess:
description: HostProcess determines if a container
should be run as a 'Host Process' container.
- This field is alpha-level and will only be
- honored by components that enable the WindowsHostProcessContainers
- feature flag. Setting this field without the
- feature flag will result in errors when validating
- the Pod. All of a Pod's containers must have
- the same effective HostProcess value (it is
- not allowed to have a mix of HostProcess containers
- and non-HostProcess containers). In addition,
+ All of a Pod's containers must have the same
+ effective HostProcess value (it is not allowed
+ to have a mix of HostProcess containers and
+ non-HostProcess containers). In addition,
if HostProcess is true then HostNetwork must
also be set to true.
type: boolean
@@ -6346,8 +6553,7 @@ spec:
type: integer
grpc:
description: GRPC specifies an action involving
- a GRPC port. This is a beta field and requires
- enabling GRPCContainerProbe feature gate.
+ a GRPC port.
properties:
port:
description: Port number of the gRPC service.
@@ -6381,7 +6587,10 @@ spec:
header to be used in HTTP probes
properties:
name:
- description: The header field name
+ description: The header field name. This
+ will be canonicalized upon output, so
+ case-variant names will be understood
+ as the same header.
type: string
value:
description: The header field value
@@ -6729,19 +6938,14 @@ spec:
namespace as this pod. \n The template will be
used to create a new ResourceClaim, which will
be bound to this pod. When this pod is deleted,
- the ResourceClaim will also be deleted. The name
- of the ResourceClaim will be -, where is the PodResourceClaim.Name.
- Pod validation will reject the pod if the concatenated
- name is not valid for a ResourceClaim (e.g. too
- long). \n An existing ResourceClaim with that
- name that is not owned by the pod will not be
- used for the pod to avoid using an unrelated resource
- by mistake. Scheduling and pod startup are then
- blocked until the unrelated ResourceClaim is removed.
- \n This field is immutable and no changes will
- be made to the corresponding ResourceClaim by
- the control plane after creating the ResourceClaim."
+ the ResourceClaim will also be deleted. The pod
+ name and resource name, along with a generated
+ component, will be used to form a unique name
+ for the ResourceClaim, which will be recorded
+ in pod.status.resourceClaimStatuses. \n This field
+ is immutable and no changes will be made to the
+ corresponding ResourceClaim by the control plane
+ after creating the ResourceClaim."
type: string
type: object
required:
@@ -6753,8 +6957,9 @@ spec:
x-kubernetes-list-type: map
restartPolicy:
description: 'Restart policy for all containers within the
- pod. One of Always, OnFailure, Never. Default to Always.
- More info: https://kubernetes.io/docs/concepts/workloads/pods/pod-lifecycle/#restart-policy'
+ pod. One of Always, OnFailure, Never. In some contexts,
+ only a subset of those values may be permitted. Default
+ to Always. More info: https://kubernetes.io/docs/concepts/workloads/pods/pod-lifecycle/#restart-policy'
type: string
runtimeClassName:
description: 'RuntimeClassName refers to a RuntimeClass object
@@ -6772,10 +6977,12 @@ spec:
type: string
schedulingGates:
description: "SchedulingGates is an opaque list of values
- that if specified will block scheduling the pod. More info:
- \ https://git.k8s.io/enhancements/keps/sig-scheduling/3521-pod-scheduling-readiness.
- \n This is an alpha-level feature enabled by PodSchedulingReadiness
- feature gate."
+ that if specified will block scheduling the pod. If schedulingGates
+ is not empty, the pod will stay in the SchedulingGated state
+ and the scheduler will not attempt to schedule the pod.
+ \n SchedulingGates can only be set at pod creation time,
+ and be removed only afterwards. \n This is a beta feature
+ enabled by the PodSchedulingReadiness feature gate."
items:
description: PodSchedulingGate is associated to a Pod to
guard its scheduling.
@@ -6885,8 +7092,9 @@ spec:
defined in a file on the node should be used. The
profile must be preconfigured on the node to work.
Must be a descending path, relative to the kubelet's
- configured seccomp profile location. Must only be
- set if type is "Localhost".
+ configured seccomp profile location. Must be set
+ if type is "Localhost". Must NOT be set for any
+ other type.
type: string
type:
description: "type indicates which kind of seccomp
@@ -6954,15 +7162,11 @@ spec:
type: string
hostProcess:
description: HostProcess determines if a container
- should be run as a 'Host Process' container. This
- field is alpha-level and will only be honored by
- components that enable the WindowsHostProcessContainers
- feature flag. Setting this field without the feature
- flag will result in errors when validating the Pod.
- All of a Pod's containers must have the same effective
+ should be run as a 'Host Process' container. All
+ of a Pod's containers must have the same effective
HostProcess value (it is not allowed to have a mix
- of HostProcess containers and non-HostProcess containers). In
- addition, if HostProcess is true then HostNetwork
+ of HostProcess containers and non-HostProcess containers).
+ In addition, if HostProcess is true then HostNetwork
must also be set to true.
type: boolean
runAsUserName:
@@ -7122,16 +7326,21 @@ spec:
requirements are ANDed.
type: object
type: object
+ x-kubernetes-map-type: atomic
matchLabelKeys:
- description: MatchLabelKeys is a set of pod label keys
+ description: "MatchLabelKeys is a set of pod label keys
to select the pods over which spreading will be calculated.
The keys are used to lookup values from the incoming
pod labels, those key-value labels are ANDed with
labelSelector to select the group of existing pods
over which spreading will be calculated for the incoming
- pod. Keys that don't exist in the incoming pod labels
- will be ignored. A null or empty list means only match
- against labelSelector.
+ pod. The same key is forbidden to exist in both MatchLabelKeys
+ and LabelSelector. MatchLabelKeys cannot be set when
+ LabelSelector isn't set. Keys that don't exist in
+ the incoming pod labels will be ignored. A null or
+ empty list means only match against labelSelector.
+ \n This is a beta field and requires the MatchLabelKeysInPodTopologySpread
+ feature gate to be enabled (enabled by default)."
items:
type: string
type: array
@@ -7391,6 +7600,7 @@ spec:
kind, uid?'
type: string
type: object
+ x-kubernetes-map-type: atomic
user:
description: 'user is optional: User is the rados
user name, default is admin More info: https://examples.k8s.io/volumes/cephfs/README.md#how-to-use-it'
@@ -7426,6 +7636,7 @@ spec:
kind, uid?'
type: string
type: object
+ x-kubernetes-map-type: atomic
volumeID:
description: 'volumeID used to identify the volume
in cinder. More info: https://examples.k8s.io/mysql-cinder-pd/README.md'
@@ -7505,6 +7716,7 @@ spec:
or its keys must be defined
type: boolean
type: object
+ x-kubernetes-map-type: atomic
csi:
description: csi (Container Storage Interface) represents
ephemeral storage that is handled by certain external
@@ -7538,6 +7750,7 @@ spec:
kind, uid?'
type: string
type: object
+ x-kubernetes-map-type: atomic
readOnly:
description: readOnly specifies a read-only configuration
for the volume. Defaults to false (read/write).
@@ -7596,6 +7809,7 @@ spec:
required:
- fieldPath
type: object
+ x-kubernetes-map-type: atomic
mode:
description: 'Optional: mode bits used to
set permissions on this file, must be an
@@ -7641,6 +7855,7 @@ spec:
required:
- resource
type: object
+ x-kubernetes-map-type: atomic
required:
- path
type: object
@@ -7668,7 +7883,7 @@ spec:
be the minimum value between the SizeLimit specified
here and the sum of memory limits of all containers
in a pod. The default is nil which means that
- the limit is undefined. More info: http://kubernetes.io/docs/user-guide/volumes#emptydir'
+ the limit is undefined. More info: https://kubernetes.io/docs/concepts/storage/volumes#emptydir'
pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$
x-kubernetes-int-or-string: true
type: object
@@ -7792,6 +8007,7 @@ spec:
- kind
- name
type: object
+ x-kubernetes-map-type: atomic
dataSourceRef:
description: 'dataSourceRef specifies the
object from which to populate the volume
@@ -7923,7 +8139,8 @@ spec:
a container, it defaults to Limits
if that is explicitly specified, otherwise
to an implementation-defined value.
- More info: https://kubernetes.io/docs/concepts/configuration/manage-resources-containers/'
+ Requests cannot exceed Limits. More
+ info: https://kubernetes.io/docs/concepts/configuration/manage-resources-containers/'
type: object
type: object
selector:
@@ -7981,6 +8198,7 @@ spec:
ANDed.
type: object
type: object
+ x-kubernetes-map-type: atomic
storageClassName:
description: 'storageClassName is the name
of the StorageClass required by the claim.
@@ -8079,6 +8297,7 @@ spec:
kind, uid?'
type: string
type: object
+ x-kubernetes-map-type: atomic
required:
- driver
type: object
@@ -8269,6 +8488,7 @@ spec:
kind, uid?'
type: string
type: object
+ x-kubernetes-map-type: atomic
targetPortal:
description: targetPortal is iSCSI Target Portal.
The Portal is either an IP or ip_addr:port if
@@ -8451,6 +8671,7 @@ spec:
the ConfigMap or its keys must be defined
type: boolean
type: object
+ x-kubernetes-map-type: atomic
downwardAPI:
description: downwardAPI information about
the downwardAPI data to project
@@ -8482,6 +8703,7 @@ spec:
required:
- fieldPath
type: object
+ x-kubernetes-map-type: atomic
mode:
description: 'Optional: mode bits
used to set permissions on this
@@ -8537,6 +8759,7 @@ spec:
required:
- resource
type: object
+ x-kubernetes-map-type: atomic
required:
- path
type: object
@@ -8608,6 +8831,7 @@ spec:
the Secret or its key must be defined
type: boolean
type: object
+ x-kubernetes-map-type: atomic
serviceAccountToken:
description: serviceAccountToken is information
about the serviceAccountToken data to project
@@ -8734,6 +8958,7 @@ spec:
kind, uid?'
type: string
type: object
+ x-kubernetes-map-type: atomic
user:
description: 'user is the rados user name. Default
is admin. More info: https://examples.k8s.io/volumes/rbd/README.md#how-to-use-it'
@@ -8778,6 +9003,7 @@ spec:
kind, uid?'
type: string
type: object
+ x-kubernetes-map-type: atomic
sslEnabled:
description: sslEnabled Flag enable/disable SSL
communication with Gateway, default false
@@ -8903,6 +9129,7 @@ spec:
kind, uid?'
type: string
type: object
+ x-kubernetes-map-type: atomic
volumeName:
description: volumeName is the human-readable name
of the StorageOS volume. Volume names are only
@@ -9076,6 +9303,7 @@ spec:
- kind
- name
type: object
+ x-kubernetes-map-type: atomic
dataSourceRef:
description: 'dataSourceRef specifies the object from which
to populate the volume with data, if a non-empty volume
@@ -9183,7 +9411,7 @@ spec:
of compute resources required. If Requests is omitted
for a container, it defaults to Limits if that is
explicitly specified, otherwise to an implementation-defined
- value. More info: https://kubernetes.io/docs/concepts/configuration/manage-resources-containers/'
+ value. Requests cannot exceed Limits. More info: https://kubernetes.io/docs/concepts/configuration/manage-resources-containers/'
type: object
type: object
selector:
@@ -9232,6 +9460,7 @@ spec:
contains only "value". The requirements are ANDed.
type: object
type: object
+ x-kubernetes-map-type: atomic
storageClassName:
description: 'storageClassName is the name of the StorageClass
required by the claim. More info: https://kubernetes.io/docs/concepts/storage/persistent-volumes#class-1'
@@ -9256,6 +9485,50 @@ spec:
items:
type: string
type: array
+ allocatedResourceStatuses:
+ additionalProperties:
+ description: When a controller receives persistentvolume
+ claim update with ClaimResourceStatus for a resource
+ that it does not recognizes, then it should ignore that
+ update and let other controllers handle it.
+ type: string
+ description: "allocatedResourceStatuses stores status of
+ resource being resized for the given PVC. Key names follow
+ standard Kubernetes label syntax. Valid values are either:
+ * Un-prefixed keys: - storage - the capacity of the volume.
+ * Custom resources must use implementation-defined prefixed
+ names such as \"example.com/my-custom-resource\" Apart
+ from above values - keys that are unprefixed or have kubernetes.io
+ prefix are considered reserved and hence may not be used.
+ \n ClaimResourceStatus can be in any of following states:
+ - ControllerResizeInProgress: State set when resize controller
+ starts resizing the volume in control-plane. - ControllerResizeFailed:
+ State set when resize has failed in resize controller
+ with a terminal error. - NodeResizePending: State set
+ when resize controller has finished resizing the volume
+ but further resizing of volume is needed on the node.
+ - NodeResizeInProgress: State set when kubelet starts
+ resizing the volume. - NodeResizeFailed: State set when
+ resizing has failed in kubelet with a terminal error.
+ Transient errors don't set NodeResizeFailed. For example:
+ if expanding a PVC for more capacity - this field can
+ be one of the following states: - pvc.status.allocatedResourceStatus['storage']
+ = \"ControllerResizeInProgress\" - pvc.status.allocatedResourceStatus['storage']
+ = \"ControllerResizeFailed\" - pvc.status.allocatedResourceStatus['storage']
+ = \"NodeResizePending\" - pvc.status.allocatedResourceStatus['storage']
+ = \"NodeResizeInProgress\" - pvc.status.allocatedResourceStatus['storage']
+ = \"NodeResizeFailed\" When this field is not set, it
+ means that no resize operation is in progress for the
+ given PVC. \n A controller that receives PVC update with
+ previously unknown resourceName or ClaimResourceStatus
+ should ignore the update for the purpose it was designed.
+ For example - a controller that only is responsible for
+ resizing capacity of the volume, should ignore PVC updates
+ that change other valid resources associated with PVC.
+ \n This is an alpha field and requires enabling RecoverVolumeExpansionFailure
+ feature."
+ type: object
+ x-kubernetes-map-type: granular
allocatedResources:
additionalProperties:
anyOf:
@@ -9263,18 +9536,30 @@ spec:
- type: string
pattern: ^(\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))(([KMGTPE]i)|[numkMGTPE]|([eE](\+|-)?(([0-9]+(\.[0-9]*)?)|(\.[0-9]+))))?$
x-kubernetes-int-or-string: true
- description: allocatedResources is the storage resource
- within AllocatedResources tracks the capacity allocated
- to a PVC. It may be larger than the actual capacity when
- a volume expansion operation is requested. For storage
- quota, the larger value from allocatedResources and PVC.spec.resources
- is used. If allocatedResources is not set, PVC.spec.resources
- alone is used for quota calculation. If a volume expansion
- capacity request is lowered, allocatedResources is only
- lowered if there are no expansion operations in progress
- and if the actual volume capacity is equal or lower than
- the requested capacity. This is an alpha field and requires
- enabling RecoverVolumeExpansionFailure feature.
+ description: "allocatedResources tracks the resources allocated
+ to a PVC including its capacity. Key names follow standard
+ Kubernetes label syntax. Valid values are either: * Un-prefixed
+ keys: - storage - the capacity of the volume. * Custom
+ resources must use implementation-defined prefixed names
+ such as \"example.com/my-custom-resource\" Apart from
+ above values - keys that are unprefixed or have kubernetes.io
+ prefix are considered reserved and hence may not be used.
+ \n Capacity reported here may be larger than the actual
+ capacity when a volume expansion operation is requested.
+ For storage quota, the larger value from allocatedResources
+ and PVC.spec.resources is used. If allocatedResources
+ is not set, PVC.spec.resources alone is used for quota
+ calculation. If a volume expansion capacity request is
+ lowered, allocatedResources is only lowered if there are
+ no expansion operations in progress and if the actual
+ volume capacity is equal or lower than the requested capacity.
+ \n A controller that receives PVC update with previously
+ unknown resourceName should ignore the update for the
+ purpose it was designed. For example - a controller that
+ only is responsible for resizing capacity of the volume,
+ should ignore PVC updates that change other valid resources
+ associated with PVC. \n This is an alpha field and requires
+ enabling RecoverVolumeExpansionFailure feature."
type: object
capacity:
additionalProperties:
@@ -9291,7 +9576,7 @@ spec:
volume claim. If underlying persistent volume is being
resized then the Condition will be set to 'ResizeStarted'.
items:
- description: PersistentVolumeClaimCondition contails details
+ description: PersistentVolumeClaimCondition contains details
about state of pvc
properties:
lastProbeTime:
@@ -9329,13 +9614,6 @@ spec:
phase:
description: phase represents the current phase of PersistentVolumeClaim.
type: string
- resizeStatus:
- description: resizeStatus stores status of resize operation.
- ResizeStatus is not set by default but when expansion
- is complete resizeStatus is set to empty string by resize
- controller or kubelet. This is an alpha field and requires
- enabling RecoverVolumeExpansionFailure feature.
- type: string
type: object
type: object
type: array
diff --git a/deploy/helm/templates/_helpers.tpl b/deploy/helm/templates/_helpers.tpl
index f2b4e59c750..6bf35fb47bc 100644
--- a/deploy/helm/templates/_helpers.tpl
+++ b/deploy/helm/templates/_helpers.tpl
@@ -283,6 +283,14 @@ TODO: For azure, we should get provider from node.Spec.ProviderID
*/}}
{{- define "kubeblocks.cloudProvider" }}
{{- $kubeVersion := .Capabilities.KubeVersion.GitVersion }}
+{{- $validProviders := .Values.validProviders}}
+{{- $provider := .Values.provider }}
+{{- $valid := false }}
+{{- range $validProviders }}
+ {{- if eq . $provider }}
+ {{- $valid = true }}
+ {{- end }}
+{{- end }}
{{- if contains "-eks" $kubeVersion }}
{{- "aws" -}}
{{- else if contains "-gke" $kubeVersion }}
@@ -293,11 +301,16 @@ TODO: For azure, we should get provider from node.Spec.ProviderID
{{- "tencentCloud" -}}
{{- else if contains "-aks" $kubeVersion }}
{{- "azure" -}}
-{{- else }}
-{{- "" -}}
+{{- else if $valid }}
+{{- $provider }}
+{{- else}}
+{{- $invalidProvider := join ", " .Values.validProviders }}
+{{- $errorMessage := printf "Warning: Your provider is invalid. Please use one of the following: %s" $invalidProvider | trimSuffix ", " }}
+{{- fail $errorMessage}}
{{- end }}
{{- end }}
+
{{/*
Define default storage class name, if cloud provider is known, specify a default storage class name.
*/}}
diff --git a/deploy/helm/templates/addons/nvidia-gpu-exporter-addon.yaml b/deploy/helm/templates/addons/nvidia-gpu-exporter-addon.yaml
index 0c4a93dd735..0302c1b35fb 100644
--- a/deploy/helm/templates/addons/nvidia-gpu-exporter-addon.yaml
+++ b/deploy/helm/templates/addons/nvidia-gpu-exporter-addon.yaml
@@ -14,7 +14,14 @@ spec:
type: Helm
helm:
- chartLocationURL: https://jihulab.com/api/v4/projects/85949/packages/helm/stable/charts/nvidia-gpu-exporter-0.3.1.tgz
+ {{- include "kubeblocks.addonChartLocationURL" ( dict "name" "nvidia-gpu-exporter" "version" "0.3.1" "values" .Values) | indent 4 }}
+ {{- include "kubeblocks.addonChartsImage" . | indent 4 }}
+
+ installOptions:
+ {{- if hasPrefix "oci://" .Values.addonChartLocationBase }}
+ version: 0.3.1
+ {{- end }}
+
installable:
autoInstall: false
diff --git a/deploy/helm/templates/addons/snapshot-controller-addon.yaml b/deploy/helm/templates/addons/snapshot-controller-addon.yaml
index 615ca462a5a..e80887f5ac1 100644
--- a/deploy/helm/templates/addons/snapshot-controller-addon.yaml
+++ b/deploy/helm/templates/addons/snapshot-controller-addon.yaml
@@ -88,4 +88,9 @@ spec:
operator: DoesNotContain
values:
- tke
- - aliyun
\ No newline at end of file
+ - aliyun
+ - key: KubeProvider
+ operator: DoesNotContain
+ values:
+ - huaweiCloud
+ - azure
\ No newline at end of file
diff --git a/deploy/helm/templates/deployment.yaml b/deploy/helm/templates/deployment.yaml
index d41a3c09da6..494b9acea5d 100644
--- a/deploy/helm/templates/deployment.yaml
+++ b/deploy/helm/templates/deployment.yaml
@@ -113,6 +113,8 @@ spec:
- name: RECOVER_VOLUME_EXPANSION_FAILURE
value: "true"
{{- end }}
+ - name: KUBE_PROVIDER
+ value: {{ .Values.provider | quote }}
{{- with .Values.securityContext }}
securityContext:
{{- toYaml . | nindent 12 }}
diff --git a/deploy/helm/templates/rbac/dataprotection_restorejob_editor_role.yaml b/deploy/helm/templates/rbac/dataprotection_restore_editor_role.yaml
similarity index 69%
rename from deploy/helm/templates/rbac/dataprotection_restorejob_editor_role.yaml
rename to deploy/helm/templates/rbac/dataprotection_restore_editor_role.yaml
index e9464749c2d..1711d4f4f70 100644
--- a/deploy/helm/templates/rbac/dataprotection_restorejob_editor_role.yaml
+++ b/deploy/helm/templates/rbac/dataprotection_restore_editor_role.yaml
@@ -1,15 +1,15 @@
-# permissions for end users to edit restorejobs.
+# permissions for end users to edit restores.
apiVersion: rbac.authorization.k8s.io/v1
kind: ClusterRole
metadata:
- name: {{ include "kubeblocks.fullname" . }}-restorejob-editor-role
+ name: {{ include "kubeblocks.fullname" . }}-restore-editor-role
labels:
{{- include "kubeblocks.labels" . | nindent 4 }}
rules:
- apiGroups:
- dataprotection.kubeblocks.io
resources:
- - restorejobs
+ - restores
verbs:
- create
- delete
@@ -21,6 +21,6 @@ rules:
- apiGroups:
- dataprotection.kubeblocks.io
resources:
- - restorejobs/status
+ - restores/status
verbs:
- get
diff --git a/deploy/helm/templates/rbac/dataprotection_restorejob_viewer_role.yaml b/deploy/helm/templates/rbac/dataprotection_restore_viewer_role.yaml
similarity index 66%
rename from deploy/helm/templates/rbac/dataprotection_restorejob_viewer_role.yaml
rename to deploy/helm/templates/rbac/dataprotection_restore_viewer_role.yaml
index ca90687a7b7..3230f6daec8 100644
--- a/deploy/helm/templates/rbac/dataprotection_restorejob_viewer_role.yaml
+++ b/deploy/helm/templates/rbac/dataprotection_restore_viewer_role.yaml
@@ -1,15 +1,15 @@
-# permissions for end users to view restorejobs.
+# permissions for end users to view restores.
apiVersion: rbac.authorization.k8s.io/v1
kind: ClusterRole
metadata:
- name: {{ include "kubeblocks.fullname" . }}-restorejob-viewer-role
+ name: {{ include "kubeblocks.fullname" . }}-restore-viewer-role
labels:
{{- include "kubeblocks.labels" . | nindent 4 }}
rules:
- apiGroups:
- dataprotection.kubeblocks.io
resources:
- - restorejobs
+ - restores
verbs:
- get
- list
@@ -17,6 +17,6 @@ rules:
- apiGroups:
- dataprotection.kubeblocks.io
resources:
- - restorejobs/status
+ - restores/status
verbs:
- get
diff --git a/deploy/helm/templates/storageclass.yaml b/deploy/helm/templates/storageclass.yaml
index 9e1d09992b7..4554a8ed784 100644
--- a/deploy/helm/templates/storageclass.yaml
+++ b/deploy/helm/templates/storageclass.yaml
@@ -12,8 +12,8 @@ metadata:
allowVolumeExpansion: true
parameters:
## parameters references: https://github.com/kubernetes-sigs/aws-ebs-csi-driver/blob/master/docs/parameters.md
- type: {{ .Values.storageClass.provider.eks.volumeType }} # io2, io1, gp3, gp2 are all SSD variant
- "csi.storage.k8s.io/fstype": {{ .Values.storageClass.provider.eks.fsType | default "xfs" }}
+ type: {{ .Values.storageClass.provider.aws.volumeType }} # io2, io1, gp3, gp2 are all SSD variant
+ "csi.storage.k8s.io/fstype": {{ .Values.storageClass.provider.aws.fsType | default "xfs" }}
provisioner: ebs.csi.aws.com
reclaimPolicy: Delete
volumeBindingMode: WaitForFirstConsumer
@@ -29,8 +29,8 @@ metadata:
allowVolumeExpansion: true
parameters:
## refer: https://github.com/kubernetes-sigs/gcp-compute-persistent-disk-csi-driver/issues/617
- type: {{ .Values.storageClass.provider.gke.volumeType }}
- csi.storage.k8s.io/fstype: {{ .Values.storageClass.provider.gke.fsType | default "xfs" }}
+ type: {{ .Values.storageClass.provider.gcp.volumeType }}
+ csi.storage.k8s.io/fstype: {{ .Values.storageClass.provider.gcp.fsType | default "xfs" }}
provisioner: pd.csi.storage.gke.io
reclaimPolicy: Delete
volumeBindingMode: WaitForFirstConsumer
@@ -62,7 +62,7 @@ metadata:
{{- include "kubeblocks.labels" . | nindent 4 }}
parameters:
## parameters references: https://cloud.tencent.com/document/product/457/44239, the fsType is not supported by tke.
- type: {{ .Values.storageClass.provider.tke.volumeType }}
+ type: {{ .Values.storageClass.provider.tencentCloud.volumeType }}
reclaimPolicy: Delete
provisioner: com.tencent.cloud.csi.cbs
volumeBindingMode: WaitForFirstConsumer
@@ -79,12 +79,31 @@ metadata:
{{- include "kubeblocks.labels" . | nindent 4 }}
parameters:
# parameters references: https://github.com/kubernetes-sigs/azuredisk-csi-driver/blob/master/docs/driver-parameters.md
- fsType: {{ .Values.storageClass.provider.aks.fsType | default "xfs" }}
- kind: {{ .Values.storageClass.provider.aks.volumeType }}
+ fsType: {{ .Values.storageClass.provider.azure.fsType | default "xfs" }}
+ kind: {{ .Values.storageClass.provider.azure.volumeType }}
skuName: Standard_LRS
provisioner: kubernetes.io/azure-disk
reclaimPolicy: Delete
volumeBindingMode: WaitForFirstConsumer
---
+{{- else if eq $cloudProvider "huaweiCloud" }} # huawei cloud
+---
+apiVersion: storage.k8s.io/v1
+kind: StorageClass
+metadata:
+ name: {{ $scName }}
+ labels:
+ {{- include "kubeblocks.labels" . | nindent 4 }}
+parameters:
+ # parameters references: https://support.huaweicloud.com/usermanual-cce/cce_10_0380.html#section3
+ csi.storage.k8s.io/csi-driver-name: disk.csi.everest.io
+ csi.storage.k8s.io/fstype: {{ .Values.storageClass.provider.huaweiCloud.fsType | default "xfs" }}
+ everest.io/disk-volume-type: {{ .Values.storageClass.provider.huaweiCloud.volumeType }}
+ everest.io/passthrough: "true"
+provisioner: everest-csi-provisioner
+reclaimPolicy: Delete
+volumeBindingMode: WaitForFirstConsumer
+allowVolumeExpansion: true
+---
{{- end }}
{{- end }}
\ No newline at end of file
diff --git a/deploy/helm/templates/storageprovider/cos.yaml b/deploy/helm/templates/storageprovider/cos.yaml
index fc9a597a87a..ec39ede7d5a 100644
--- a/deploy/helm/templates/storageprovider/cos.yaml
+++ b/deploy/helm/templates/storageprovider/cos.yaml
@@ -33,6 +33,16 @@ spec:
csi.storage.k8s.io/node-publish-secret-name: {{ `{{ .CSIDriverSecretRef.Name }}` }}
csi.storage.k8s.io/node-publish-secret-namespace: {{ `{{ .CSIDriverSecretRef.Namespace }}` }}
+ datasafedConfigTemplate: |
+ [storage]
+ type = s3
+ provider = TencentCOS
+ env_auth = false
+ access_key_id = {{ `{{ index .Parameters "accessKeyId" }}` }}
+ secret_access_key = {{ `{{ index .Parameters "secretAccessKey" }}` }}
+ endpoint = {{ `{{ printf "cos.%s.myqcloud.com" .Parameters.region }}` }}
+ root = {{ `{{ index .Parameters "bucket" }}` }}
+
parametersSchema:
openAPIV3Schema:
type: "object"
diff --git a/deploy/helm/templates/storageprovider/gcs.yaml b/deploy/helm/templates/storageprovider/gcs.yaml
index 9e835fc4f24..0d8ae56f63b 100644
--- a/deploy/helm/templates/storageprovider/gcs.yaml
+++ b/deploy/helm/templates/storageprovider/gcs.yaml
@@ -8,13 +8,12 @@ metadata:
spec:
csiDriverName: ru.yandex.s3.csi
csiDriverSecretTemplate: |
- accessKeyID: {{ `{{ index .Parameters "accessKeyId" }}` }}
- secretAccessKey: {{ `{{ index .Parameters "secretAccessKey" }}` }}
- {{ `{{- $region := index .Parameters "region" }}` }}
{{ `{{- $endpoint := index .Parameters "endpoint" }}` }}
{{ `{{- if not $endpoint }}` }}
{{ `{{- $endpoint = (printf "https://storage.googleapis.com") }}` }}
{{ `{{- end }}` }}
+ accessKeyID: {{ `{{ index .Parameters "accessKeyId" }}` }}
+ secretAccessKey: {{ `{{ index .Parameters "secretAccessKey" }}` }}
endpoint: {{ `{{ $endpoint }}` }}
storageClassTemplate: |
@@ -33,6 +32,20 @@ spec:
csi.storage.k8s.io/node-publish-secret-name: {{ `{{ .CSIDriverSecretRef.Name }}` }}
csi.storage.k8s.io/node-publish-secret-namespace: {{ `{{ .CSIDriverSecretRef.Namespace }}` }}
+ datasafedConfigTemplate: |
+ [storage]
+ type = s3
+ provider = GCS
+ env_auth = false
+ access_key_id = {{ `{{ index .Parameters "accessKeyId" }}` }}
+ secret_access_key = {{ `{{ index .Parameters "secretAccessKey" }}` }}
+ {{ `{{- $endpoint := index .Parameters "endpoint" }}` }}
+ {{ `{{- if not $endpoint }}` }}
+ {{ `{{- $endpoint = (printf "https://storage.googleapis.com") }}` }}
+ {{ `{{- end }}` }}
+ endpoint = {{ `{{ $endpoint }}` }}
+ root = {{ `{{ index .Parameters "bucket" }}` }}
+
parametersSchema:
openAPIV3Schema:
type: "object"
diff --git a/deploy/helm/templates/storageprovider/minio.yaml b/deploy/helm/templates/storageprovider/minio.yaml
index 27e111babca..62ff8b65489 100644
--- a/deploy/helm/templates/storageprovider/minio.yaml
+++ b/deploy/helm/templates/storageprovider/minio.yaml
@@ -27,6 +27,16 @@ spec:
csi.storage.k8s.io/node-publish-secret-name: {{ `{{ .CSIDriverSecretRef.Name }}` }}
csi.storage.k8s.io/node-publish-secret-namespace: {{ `{{ .CSIDriverSecretRef.Namespace }}` }}
+ datasafedConfigTemplate: |
+ [storage]
+ type = s3
+ provider = Minio
+ env_auth = false
+ access_key_id = {{ `{{ index .Parameters "accessKeyId" }}` }}
+ secret_access_key = {{ `{{ index .Parameters "secretAccessKey" }}` }}
+ endpoint = {{ `{{ index .Parameters "endpoint" }}` }}
+ root = {{ `{{ index .Parameters "bucket" }}` }}
+
parametersSchema:
openAPIV3Schema:
type: "object"
diff --git a/deploy/helm/templates/storageprovider/obs.yaml b/deploy/helm/templates/storageprovider/obs.yaml
index 9b5027a7a90..a61ea97b19a 100644
--- a/deploy/helm/templates/storageprovider/obs.yaml
+++ b/deploy/helm/templates/storageprovider/obs.yaml
@@ -33,6 +33,17 @@ spec:
csi.storage.k8s.io/node-publish-secret-name: {{ `{{ .CSIDriverSecretRef.Name }}` }}
csi.storage.k8s.io/node-publish-secret-namespace: {{ `{{ .CSIDriverSecretRef.Namespace }}` }}
+ datasafedConfigTemplate: |
+ [storage]
+ type = s3
+ provider = HuaweiOBS
+ env_auth = false
+ access_key_id = {{ `{{ index .Parameters "accessKeyId" }}` }}
+ secret_access_key = {{ `{{ index .Parameters "secretAccessKey" }}` }}
+ region = {{ `{{ index .Parameters "region" }}` }}
+ endpoint = {{ `{{ printf "obs.%s.myhuaweicloud.com" .Parameters.region }}` }}
+ root = {{ `{{ index .Parameters "bucket" }}` }}
+
parametersSchema:
openAPIV3Schema:
type: "object"
diff --git a/deploy/helm/templates/storageprovider/oss.yaml b/deploy/helm/templates/storageprovider/oss.yaml
index 05f41d81b36..5ff3b780e95 100644
--- a/deploy/helm/templates/storageprovider/oss.yaml
+++ b/deploy/helm/templates/storageprovider/oss.yaml
@@ -32,6 +32,16 @@ spec:
csi.storage.k8s.io/node-publish-secret-name: {{ `{{ .CSIDriverSecretRef.Name }}` }}
csi.storage.k8s.io/node-publish-secret-namespace: {{ `{{ .CSIDriverSecretRef.Namespace }}` }}
+ datasafedConfigTemplate: |
+ [storage]
+ type = s3
+ provider = Alibaba
+ env_auth = false
+ access_key_id = {{ `{{ index .Parameters "accessKeyId" }}` }}
+ secret_access_key = {{ `{{ index .Parameters "secretAccessKey" }}` }}
+ endpoint = {{ `{{- printf "oss-%s.aliyuncs.com" .Parameters.region) }}` }}
+ root = {{ `{{ index .Parameters "bucket" }}` }}
+
parametersSchema:
openAPIV3Schema:
type: "object"
diff --git a/deploy/helm/templates/storageprovider/pvc.yaml b/deploy/helm/templates/storageprovider/pvc.yaml
new file mode 100644
index 00000000000..582f26697b1
--- /dev/null
+++ b/deploy/helm/templates/storageprovider/pvc.yaml
@@ -0,0 +1,32 @@
+apiVersion: storage.kubeblocks.io/v1alpha1
+kind: StorageProvider
+metadata:
+ name: pvc
+ labels:
+ {{- include "kubeblocks.labels" . | nindent 4 }}
+spec:
+ persistentVolumeClaimTemplate: |
+ spec:
+ {{- $scName := (include "kubeblocks.defaultStorageClass" .) }}
+ storageClassName: {{ printf `{{ .Parameters.storageClassName | default %q }}` $scName }}
+ accessModes:
+ - {{ `{{ .Parameters.accessMode | default "ReadWriteOnce" }}` }}
+ volumeMode: {{ `{{ .Parameters.volumeMode | default "Filesystem" }}` }}
+
+ parametersSchema:
+ openAPIV3Schema:
+ type: "object"
+ properties:
+ storageClassName:
+ type: string
+ description: "the name of the StorageClass used to create the PVC"
+ accessMode:
+ type: string
+ description: "the access mode used to create the PVC"
+ default: "ReadWriteOnce"
+ enum: ["ReadWriteOnce", "ReadWriteMany", "ReadWriteOncePod"]
+ volumeMode:
+ type: string
+ description: "the volume mode used to create the PVC"
+ default: "Filesystem"
+ enum: ["Filesystem", "Block"]
\ No newline at end of file
diff --git a/deploy/helm/templates/storageprovider/s3.yaml b/deploy/helm/templates/storageprovider/s3.yaml
index cdc85e12b69..8bc6730d1b6 100644
--- a/deploy/helm/templates/storageprovider/s3.yaml
+++ b/deploy/helm/templates/storageprovider/s3.yaml
@@ -36,6 +36,17 @@ spec:
csi.storage.k8s.io/node-publish-secret-name: {{ `{{ .CSIDriverSecretRef.Name }}` }}
csi.storage.k8s.io/node-publish-secret-namespace: {{ `{{ .CSIDriverSecretRef.Namespace }}` }}
+ datasafedConfigTemplate: |
+ [storage]
+ type = s3
+ provider = AWS
+ env_auth = false
+ access_key_id = {{ `{{ index .Parameters "accessKeyId" }}` }}
+ secret_access_key = {{ `{{ index .Parameters "secretAccessKey" }}` }}
+ region = {{ `{{ index .Parameters "region" }}` }}
+ endpoint = {{ `{{ index .Parameters "endpoint" }}` }}
+ root = {{ `{{ index .Parameters "bucket" }}` }}
+
parametersSchema:
openAPIV3Schema:
type: "object"
diff --git a/deploy/helm/values.yaml b/deploy/helm/values.yaml
index 5ca5f110eb7..2faf409e0ed 100644
--- a/deploy/helm/values.yaml
+++ b/deploy/helm/values.yaml
@@ -217,7 +217,7 @@ resources: {}
# memory: 128Mi
# requests:
# cpu: 10m
- # memory: 64Mi
+# memory: 64Mi
## @param priorityClassName
##
@@ -246,10 +246,10 @@ nodeSelector: {}
## @param tolerations
##
tolerations:
- - key: kb-controller
- operator: Equal
- value: "true"
- effect: NoSchedule
+- key: kb-controller
+ operator: Equal
+ value: "true"
+ effect: NoSchedule
## @param affinity
@@ -257,33 +257,33 @@ tolerations:
affinity:
nodeAffinity:
preferredDuringSchedulingIgnoredDuringExecution:
- - weight: 100
- preference:
- matchExpressions:
- - key: kb-controller
- operator: In
- values:
- - "true"
+ - weight: 100
+ preference:
+ matchExpressions:
+ - key: kb-controller
+ operator: In
+ values:
+ - "true"
## @param data plane settings
##
dataPlane:
tolerations:
- - key: kb-data
- operator: Equal
- value: "true"
- effect: NoSchedule
+ - key: kb-data
+ operator: Equal
+ value: "true"
+ effect: NoSchedule
affinity:
nodeAffinity:
preferredDuringSchedulingIgnoredDuringExecution:
- - weight: 100
- preference:
- matchExpressions:
- - key: kb-data
- operator: In
- values:
- - "true"
+ - weight: 100
+ preference:
+ matchExpressions:
+ - key: kb-data
+ operator: In
+ values:
+ - "true"
## AdmissionWebhooks settings
##
@@ -321,7 +321,7 @@ dataProtection:
pullPolicy: IfNotPresent
# Overrides the image tag whose default is the chart appVersion.
tag: ""
- imagePullSecrets: [ ]
+ imagePullSecrets: []
## BackupRepo settings
##
@@ -346,10 +346,10 @@ backupRepo:
accessKeyId: ""
secretAccessKey: ""
-## Addon controller settings, this will require cluster-admin clusterrole.
-##
-## @param addonController.enabled
-## @param addonController.jobTTL - is addon job time-to-live period, this value is time.Duration-parseable string.
+ ## Addon controller settings, this will require cluster-admin clusterrole.
+ ##
+ ## @param addonController.enabled
+ ## @param addonController.jobTTL - is addon job time-to-live period, this value is time.Duration-parseable string.
## default value is "5m" if not provided.
## @param addonController.jobImagePullPolicy - addon install job image pull policy.
addonController:
@@ -377,10 +377,10 @@ addonChartsImage:
## @param addonHelmInstallOptions - addon helm install options.
addonHelmInstallOptions:
- - "--atomic"
- - "--cleanup-on-fail"
- - "--wait"
- - "--insecure-skip-tls-verify"
+- "--atomic"
+- "--cleanup-on-fail"
+- "--wait"
+- "--insecure-skip-tls-verify"
## Prometheus Addon
##
@@ -410,21 +410,21 @@ prometheus:
## Ref: https://kubernetes.io/docs/concepts/configuration/assign-pod-node/
##
tolerations:
- - key: kb-controller
- operator: Equal
- value: "true"
- effect: NoSchedule
+ - key: kb-controller
+ operator: Equal
+ value: "true"
+ effect: NoSchedule
affinity:
nodeAffinity:
preferredDuringSchedulingIgnoredDuringExecution:
- - weight: 100
- preference:
- matchExpressions:
- - key: kb-controller
- operator: In
- values:
- - "true"
+ - weight: 100
+ preference:
+ matchExpressions:
+ - key: kb-controller
+ operator: In
+ values:
+ - "true"
persistentVolume:
## If true, alertmanager will create/use a Persistent Volume Claim
@@ -471,7 +471,7 @@ prometheus:
# memory: 32Mi
# requests:
# cpu: 10m
- # memory: 32Mi
+ # memory: 32Mi
## Security context to be added to alertmanager pods
##
@@ -607,8 +607,8 @@ prometheus:
## Additional Prometheus server container flags
##
extraFlags:
- - web.enable-lifecycle
- - web.enable-remote-write-receiver
+ - web.enable-lifecycle
+ - web.enable-remote-write-receiver
## Additional Prometheus server container arguments
##
@@ -631,21 +631,21 @@ prometheus:
## Ref: https://kubernetes.io/docs/concepts/configuration/assign-pod-node/
##
tolerations:
- - key: kb-controller
- operator: Equal
- value: "true"
- effect: NoSchedule
+ - key: kb-controller
+ operator: Equal
+ value: "true"
+ effect: NoSchedule
affinity:
nodeAffinity:
preferredDuringSchedulingIgnoredDuringExecution:
- - weight: 100
- preference:
- matchExpressions:
- - key: kb-controller
- operator: In
- values:
- - "true"
+ - weight: 100
+ preference:
+ matchExpressions:
+ - key: kb-controller
+ operator: In
+ values:
+ - "true"
persistentVolume:
## If true, Prometheus server will create/use a Persistent Volume Claim
@@ -685,7 +685,7 @@ prometheus:
# memory: 512Mi
# requests:
# cpu: 500m
- # memory: 512Mi
+ # memory: 512Mi
## Prometheus' data retention period (default if not specified is 15 days)
##
@@ -790,10 +790,10 @@ prometheus:
##
alertmanagerFiles:
alertmanager.yml:
- global: { }
+ global: {}
receivers:
- - name: default-receiver
+ - name: default-receiver
route:
receiver: default-receiver
@@ -1306,117 +1306,117 @@ prometheus:
serverFiles:
prometheus.yml:
rule_files:
- - /etc/config/recording_rules.yml
- - /etc/config/alerting_rules.yml
- - /etc/config/kubelet_alert_rules.yml
- - /etc/config/mysql_alert_rules.yml
- - /etc/config/postgresql_alert_rules.yml
- - /etc/config/redis_alert_rules.yml
- - /etc/config/kafka_alert_rules.yml
- - /etc/config/mongodb_alert_rules.yml
+ - /etc/config/recording_rules.yml
+ - /etc/config/alerting_rules.yml
+ - /etc/config/kubelet_alert_rules.yml
+ - /etc/config/mysql_alert_rules.yml
+ - /etc/config/postgresql_alert_rules.yml
+ - /etc/config/redis_alert_rules.yml
+ - /etc/config/kafka_alert_rules.yml
+ - /etc/config/mongodb_alert_rules.yml
scrape_configs:
- - job_name: prometheus
- static_configs:
- - targets:
- - localhost:9090
-
- # Scrape config for kubeblocks managed service endpoints.
- #
- # The relabeling allows the actual service scrape endpoint to be configured
- # via the following annotations:
- #
- # * `monitor.kubeblocks.io/scrape`: Only scrape services that have a value of
- # `true`.
- # * `monitor.kubeblocks.io/scheme`: If the metrics endpoint is secured then you will need
- # to set this to `https` & most likely set the `tls_config` of the scrape config.
- # * `monitor.kubeblocks.io/path`: If the metrics path is not `/metrics` override this.
- # * `monitor.kubeblocks.io/port`: If the metrics are exposed on a different port to the
- # service then set this appropriately.
- # * `monitor.kubeblocks.io/param_`: If the metrics endpoint uses parameters
- # then you can set any parameter
- - job_name: 'kubeblocks-service'
- honor_labels: true
-
- kubernetes_sd_configs:
- - role: endpoints
-
- relabel_configs:
- - source_labels: [ __meta_kubernetes_service_label_app_kubernetes_io_managed_by ]
- action: keep
- regex: kubeblocks
- - source_labels: [ __meta_kubernetes_service_label_monitor_kubeblocks_io_managed_by ]
- action: drop
- regex: agamotto
- - source_labels: [ __meta_kubernetes_service_annotation_monitor_kubeblocks_io_scrape ]
- action: keep
- regex: true
- - source_labels: [ __meta_kubernetes_service_annotation_monitor_kubeblocks_io_scheme ]
- action: replace
- target_label: __scheme__
- regex: (https?)
- - source_labels: [ __meta_kubernetes_service_annotation_monitor_kubeblocks_io_path ]
- action: replace
- target_label: __metrics_path__
- regex: (.+)
- - source_labels: [ __address__, __meta_kubernetes_service_annotation_monitor_kubeblocks_io_port ]
- action: replace
- target_label: __address__
- regex: (.+?)(?::\d+)?;(\d+)
- replacement: $1:$2
- - action: labelmap
- regex: __meta_kubernetes_service_annotation_monitor_kubeblocks_io_param_(.+)
- replacement: __param_$1
- - action: labelmap
- regex: __meta_kubernetes_service_label_(.+)
- - source_labels: [ __meta_kubernetes_namespace ]
- action: replace
- target_label: namespace
- - source_labels: [ __meta_kubernetes_service_name ]
- action: replace
- target_label: service
- - source_labels: [ __meta_kubernetes_pod_node_name ]
- action: replace
- target_label: node
- - source_labels: [ __meta_kubernetes_pod_name ]
- action: replace
- target_label: pod
- - source_labels: [ __meta_kubernetes_pod_phase ]
- regex: Pending|Succeeded|Failed|Completed
- action: drop
-
- - job_name: 'kubeblocks-agamotto'
- honor_labels: true
-
- kubernetes_sd_configs:
- - role: endpoints
-
- relabel_configs:
- - source_labels: [ __meta_kubernetes_service_label_monitor_kubeblocks_io_managed_by ]
- action: keep
- regex: agamotto
- - source_labels: [ __meta_kubernetes_service_annotation_monitor_kubeblocks_io_scrape ]
- action: keep
- regex: true
- - source_labels: [ __meta_kubernetes_service_annotation_monitor_kubeblocks_io_scheme ]
- action: replace
- target_label: __scheme__
- regex: (https?)
- - source_labels: [ __meta_kubernetes_service_annotation_monitor_kubeblocks_io_path ]
- action: replace
- target_label: __metrics_path__
- regex: (.+)
- - source_labels: [ __address__, __meta_kubernetes_service_annotation_monitor_kubeblocks_io_port ]
- action: replace
- target_label: __address__
- regex: (.+?)(?::\d+)?;(\d+)
- replacement: $1:$2
- - action: labelmap
- regex: __meta_kubernetes_service_annotation_monitor_kubeblocks_io_param_(.+)
- replacement: __param_$1
- - source_labels: [ __meta_kubernetes_pod_phase ]
- regex: Pending|Succeeded|Failed|Completed
- action: drop
+ - job_name: prometheus
+ static_configs:
+ - targets:
+ - localhost:9090
+
+ # Scrape config for kubeblocks managed service endpoints.
+ #
+ # The relabeling allows the actual service scrape endpoint to be configured
+ # via the following annotations:
+ #
+ # * `monitor.kubeblocks.io/scrape`: Only scrape services that have a value of
+ # `true`.
+ # * `monitor.kubeblocks.io/scheme`: If the metrics endpoint is secured then you will need
+ # to set this to `https` & most likely set the `tls_config` of the scrape config.
+ # * `monitor.kubeblocks.io/path`: If the metrics path is not `/metrics` override this.
+ # * `monitor.kubeblocks.io/port`: If the metrics are exposed on a different port to the
+ # service then set this appropriately.
+ # * `monitor.kubeblocks.io/param_`: If the metrics endpoint uses parameters
+ # then you can set any parameter
+ - job_name: 'kubeblocks-service'
+ honor_labels: true
+
+ kubernetes_sd_configs:
+ - role: endpoints
+
+ relabel_configs:
+ - source_labels: [__meta_kubernetes_service_label_app_kubernetes_io_managed_by]
+ action: keep
+ regex: kubeblocks
+ - source_labels: [__meta_kubernetes_service_label_monitor_kubeblocks_io_managed_by]
+ action: drop
+ regex: agamotto
+ - source_labels: [__meta_kubernetes_service_annotation_monitor_kubeblocks_io_scrape]
+ action: keep
+ regex: true
+ - source_labels: [__meta_kubernetes_service_annotation_monitor_kubeblocks_io_scheme]
+ action: replace
+ target_label: __scheme__
+ regex: (https?)
+ - source_labels: [__meta_kubernetes_service_annotation_monitor_kubeblocks_io_path]
+ action: replace
+ target_label: __metrics_path__
+ regex: (.+)
+ - source_labels: [__address__, __meta_kubernetes_service_annotation_monitor_kubeblocks_io_port]
+ action: replace
+ target_label: __address__
+ regex: (.+?)(?::\d+)?;(\d+)
+ replacement: $1:$2
+ - action: labelmap
+ regex: __meta_kubernetes_service_annotation_monitor_kubeblocks_io_param_(.+)
+ replacement: __param_$1
+ - action: labelmap
+ regex: __meta_kubernetes_service_label_(.+)
+ - source_labels: [__meta_kubernetes_namespace]
+ action: replace
+ target_label: namespace
+ - source_labels: [__meta_kubernetes_service_name]
+ action: replace
+ target_label: service
+ - source_labels: [__meta_kubernetes_pod_node_name]
+ action: replace
+ target_label: node
+ - source_labels: [__meta_kubernetes_pod_name]
+ action: replace
+ target_label: pod
+ - source_labels: [__meta_kubernetes_pod_phase]
+ regex: Pending|Succeeded|Failed|Completed
+ action: drop
+
+ - job_name: 'kubeblocks-agamotto'
+ honor_labels: true
+
+ kubernetes_sd_configs:
+ - role: endpoints
+
+ relabel_configs:
+ - source_labels: [__meta_kubernetes_service_label_monitor_kubeblocks_io_managed_by]
+ action: keep
+ regex: agamotto
+ - source_labels: [__meta_kubernetes_service_annotation_monitor_kubeblocks_io_scrape]
+ action: keep
+ regex: true
+ - source_labels: [__meta_kubernetes_service_annotation_monitor_kubeblocks_io_scheme]
+ action: replace
+ target_label: __scheme__
+ regex: (https?)
+ - source_labels: [__meta_kubernetes_service_annotation_monitor_kubeblocks_io_path]
+ action: replace
+ target_label: __metrics_path__
+ regex: (.+)
+ - source_labels: [__address__, __meta_kubernetes_service_annotation_monitor_kubeblocks_io_port]
+ action: replace
+ target_label: __address__
+ regex: (.+?)(?::\d+)?;(\d+)
+ replacement: $1:$2
+ - action: labelmap
+ regex: __meta_kubernetes_service_annotation_monitor_kubeblocks_io_param_(.+)
+ replacement: __param_$1
+ - source_labels: [__meta_kubernetes_pod_phase]
+ regex: Pending|Succeeded|Failed|Completed
+ action: drop
pushgateway:
## If false, pushgateway will not be installed
@@ -1489,27 +1489,27 @@ grafana:
# memory: 128Mi
# requests:
# cpu: 100m
- # memory: 128Mi
+ # memory: 128Mi
## Node tolerations for grafana scheduling to nodes with taints
## Ref: https://kubernetes.io/docs/concepts/configuration/assign-pod-node/
##
tolerations:
- - key: kb-controller
- operator: Equal
- value: "true"
- effect: NoSchedule
+ - key: kb-controller
+ operator: Equal
+ value: "true"
+ effect: NoSchedule
affinity:
nodeAffinity:
preferredDuringSchedulingIgnoredDuringExecution:
- - weight: 100
- preference:
- matchExpressions:
- - key: kb-controller
- operator: In
- values:
- - "true"
+ - weight: 100
+ preference:
+ matchExpressions:
+ - key: kb-controller
+ operator: In
+ values:
+ - "true"
## Timezone for the default dashboards
## Other options are: browser or a specific timezone, i.e. Europe/Luxembourg
@@ -1564,7 +1564,7 @@ grafana:
# ingressClassName: nginx
# Values can be templated
annotations: {}
- # kubernetes.io/ingress.class: nginx
+ # kubernetes.io/ingress.class: nginx
# kubernetes.io/tls-acme: "true"
labels: {}
path: /
@@ -1573,7 +1573,7 @@ grafana:
pathType: Prefix
hosts:
- - chart-example.local
+ - chart-example.local
## Extra paths to prepend to every host configuration. This is useful when working with annotation based services.
extraPaths: []
# - path: /*
@@ -1631,26 +1631,26 @@ snapshot-controller:
tag: v6.2.1
tolerations:
- - key: kb-controller
- operator: Equal
- value: "true"
- effect: NoSchedule
+ - key: kb-controller
+ operator: Equal
+ value: "true"
+ effect: NoSchedule
volumeSnapshotClasses:
- - name: default-vsc
- driver: hostpath.csi.k8s.io
- deletionPolicy: Delete
+ - name: default-vsc
+ driver: hostpath.csi.k8s.io
+ deletionPolicy: Delete
affinity:
nodeAffinity:
preferredDuringSchedulingIgnoredDuringExecution:
- - weight: 100
- preference:
- matchExpressions:
- - key: kb-controller
- operator: In
- values:
- - "true"
+ - weight: 100
+ preference:
+ matchExpressions:
+ - key: kb-controller
+ operator: In
+ values:
+ - "true"
kubeblocks-csi-driver:
enabled: false
@@ -1690,13 +1690,13 @@ alertmanager-webhook-adaptor:
affinity:
nodeAffinity:
preferredDuringSchedulingIgnoredDuringExecution:
- - weight: 100
- preference:
- matchExpressions:
- - key: kb-controller
- operator: In
- values:
- - "true"
+ - weight: 100
+ preference:
+ matchExpressions:
+ - key: kb-controller
+ operator: In
+ values:
+ - "true"
## ConfigMap override where fullname is {{.Release.Name}}-{{.Values.configMapOverrideName}}
##
@@ -1704,7 +1704,7 @@ alertmanager-webhook-adaptor:
## Webhook-Adaptor ConfigMap Entries
configFiles:
- config.yaml: { }
+ config.yaml: {}
csi-hostpath-driver:
## @param csi-hostpath-driver.enabled -- Enable csi-hostpath-driver chart.
@@ -1754,36 +1754,46 @@ agamotto:
registry: registry.cn-hangzhou.aliyuncs.com
+provider: "" # cloud be "aws","gcp","aliyun","tencentCloud", "huaweiCloud", "azure"
+validProviders:
+ - "aws"
+ - "gcp"
+ - "aliyun"
+ - "tencentCloud"
+ - "huaweiCloud"
+ - "azure"
+ - ""
## @section KubeBlocks default storageClass Parameters for cloud provider.
storageClass:
## @param storageClass.name -- Specifies the name of the default storage class.
## If name is not specified and KubeBlocks deployed in a cloud, a default name will be generated.
##
name: ""
-
## @param storageClass.create -- Specifies whether the storage class should be created. If storageClass.name is not
## specified or generated, this value will be ignored.
##
create: true
-
mountOptions:
- - noatime
- - nobarrier
+ - noatime
+ - nobarrier
provider:
- eks:
+ aws:
volumeType: gp3
fsType: xfs
- gke:
+ gcp:
volumeType: pd-balanced
fsType: xfs
aliyun:
volumeType: cloud_essd
fsType: xfs
- aks:
+ azure:
volumeType: managed
fsType: xfs
- tke:
+ tencentCloud:
volumeType: CLOUD_SSD
+ huaweiCloud: # Huawei Cloud
+ volumeType: SSD
+ fsType: ext4
external-dns:
enabled: false
@@ -1793,4 +1803,3 @@ external-dns:
operator: Equal
value: "true"
effect: NoSchedule
-
diff --git a/deploy/milvus/templates/backuppolicytemplate.yaml b/deploy/milvus/templates/backuppolicytemplate.yaml
index c1be0359c64..3f91dcd599d 100644
--- a/deploy/milvus/templates/backuppolicytemplate.yaml
+++ b/deploy/milvus/templates/backuppolicytemplate.yaml
@@ -9,14 +9,14 @@ spec:
clusterDefinitionRef: milvus
backupPolicies:
- componentDefRef: milvus
- retention:
- ttl: 7d
- schedule:
- snapshot:
- enable: false
- cronExpression: "0 18 * * 0"
- snapshot:
- target:
- connectionCredentialKey:
- passwordKey: password
- usernameKey: username
\ No newline at end of file
+ retentionPeriod: 7d
+ backupMethods:
+ - name: volume-snapshot
+ snapshotVolumes: true
+ targetVolumes:
+ volumes:
+ - data
+ schedules:
+ - backupMethod: volume-snapshot
+ enabled: false
+ cronExpression: "0 18 * * 0"
\ No newline at end of file
diff --git a/deploy/mongodb/dataprotection/backup-info-collector.sh b/deploy/mongodb/dataprotection/backup-info-collector.sh
index 8322366235f..e3d2b63d4d3 100644
--- a/deploy/mongodb/dataprotection/backup-info-collector.sh
+++ b/deploy/mongodb/dataprotection/backup-info-collector.sh
@@ -1,6 +1,6 @@
function get_current_time() {
CLIENT=`which mongosh>/dev/null&&echo mongosh||echo mongo`
- curr_time=$(${CLIENT} -u ${DB_USER} -p ${DB_PASSWORD} --port 27017 --host ${DB_HOST} --authenticationDatabase admin --eval 'db.isMaster().lastWrite.lastWriteDate.getTime()/1000' --quiet)
+ curr_time=$(${CLIENT} -u ${DP_DB_USER} -p ${DP_DB_PASSWORD} --port 27017 --host ${DP_DB_HOST} --authenticationDatabase admin --eval 'db.isMaster().lastWrite.lastWriteDate.getTime()/1000' --quiet)
curr_time=$(date -d "@${curr_time}" -u '+%Y-%m-%dT%H:%M:%SZ')
echo $curr_time
}
@@ -11,6 +11,6 @@ function stat_and_save_backup_info() {
if [ -z $STOP_TIME ]; then
STOP_TIME=`get_current_time`
fi
- TOTAL_SIZE=$(du -shx ${BACKUP_DIR}|awk '{print $1}')
- echo "{\"totalSize\":\"$TOTAL_SIZE\",\"manifests\":{\"backupLog\":{\"startTime\":\"${START_TIME}\",\"stopTime\":\"${STOP_TIME}\"},\"backupTool\":{\"uploadTotalSize\":\"${TOTAL_SIZE}\"}}}" > ${BACKUP_DIR}/backup.info
+ TOTAL_SIZE=$(du -shx ${DP_BACKUP_DIR}|awk '{print $1}')
+ echo "{\"totalSize\":\"$TOTAL_SIZE\",\"timeRange\":{\"start\":\"${START_TIME}\",\"end\":\"${STOP_TIME}\"}}" > ${DP_BACKUP_DIR}/backup.info
}
\ No newline at end of file
diff --git a/deploy/mongodb/dataprotection/datafile-backup.sh b/deploy/mongodb/dataprotection/datafile-backup.sh
new file mode 100644
index 00000000000..d5eee06e666
--- /dev/null
+++ b/deploy/mongodb/dataprotection/datafile-backup.sh
@@ -0,0 +1,10 @@
+if [ -d ${DP_BACKUP_DIR} ]; then
+ rm -rf ${DP_BACKUP_DIR}
+fi
+mkdir -p ${DP_BACKUP_DIR} && cd ${DATA_DIR}
+START_TIME=`get_current_time`
+# TODO: flush data and locked write, otherwise data maybe inconsistent
+tar -czvf ${DP_BACKUP_DIR}/${DP_BACKUP_NAME}.tar.gz ./
+rm -rf mongodb.backup
+# stat and save the backup information
+stat_and_save_backup_info $START_TIME
\ No newline at end of file
diff --git a/deploy/mongodb/dataprotection/datafile-restore.sh b/deploy/mongodb/dataprotection/datafile-restore.sh
new file mode 100644
index 00000000000..a527c4b9528
--- /dev/null
+++ b/deploy/mongodb/dataprotection/datafile-restore.sh
@@ -0,0 +1,12 @@
+set -e
+mkdir -p ${DATA_DIR}
+res=`ls -A ${DATA_DIR}`
+data_protection_file=${DATA_DIR}/.kb-data-protection
+if [ ! -z "${res}" ] && [ ! -f ${data_protection_file} ]; then
+ echo "${DATA_DIR} is not empty! Please make sure that the directory is empty before restoring the backup."
+ exit 1
+fi
+cd ${DATA_DIR} && touch mongodb.backup
+touch ${data_protection_file}
+tar -xvf ${DP_BACKUP_DIR}/${DP_BACKUP_NAME}.tar.gz -C ${DATA_DIR}
+rm -rf ${data_protection_file} && sync
\ No newline at end of file
diff --git a/deploy/mongodb/dataprotection/mongodump-backup.sh b/deploy/mongodb/dataprotection/mongodump-backup.sh
new file mode 100644
index 00000000000..0bae45552c2
--- /dev/null
+++ b/deploy/mongodb/dataprotection/mongodump-backup.sh
@@ -0,0 +1,12 @@
+if [ -d ${DP_BACKUP_DIR} ]; then
+ rm -rf ${DP_BACKUP_DIR}
+fi
+mkdir -p ${DP_BACKUP_DIR}
+
+# TODO: support endpoint env for sharding cluster.
+mongo_uri="mongodb://${DP_DB_HOST}:${DP_DB_PORT}"
+START_TIME=`get_current_time`
+mongodump --uri ${mongo_uri} -u ${DP_DB_USER} -p ${DP_DB_PASSWORD} --authenticationDatabase admin --out ${DP_BACKUP_DIR}
+
+# stat and save the backup information
+stat_and_save_backup_info $START_TIME
\ No newline at end of file
diff --git a/deploy/mongodb/dataprotection/mongodump-restore.sh b/deploy/mongodb/dataprotection/mongodump-restore.sh
new file mode 100644
index 00000000000..66ebfe5b4f2
--- /dev/null
+++ b/deploy/mongodb/dataprotection/mongodump-restore.sh
@@ -0,0 +1,6 @@
+mongo_uri="mongodb://${DP_DB_HOST}:${DP_DB_PORT}"
+for dir_name in $(ls ${DP_BACKUP_DIR} -l | grep ^d | awk '{print $9}'); do
+ database_dir=${DP_BACKUP_DIR}/$dir_name
+ echo "INFO: restoring from ${database_dir}"
+ mongorestore --uri ${mongo_uri} -u ${MONGODB_ROOT_USER} -p ${MONGODB_ROOT_PASSWORD} -d $dir_name --authenticationDatabase admin ${database_dir}
+done
\ No newline at end of file
diff --git a/deploy/mongodb/dataprotection/pitr-backup.sh b/deploy/mongodb/dataprotection/pitr-backup.sh
index 211b16c0033..eb9cfd80e2d 100644
--- a/deploy/mongodb/dataprotection/pitr-backup.sh
+++ b/deploy/mongodb/dataprotection/pitr-backup.sh
@@ -1,12 +1,12 @@
#!/bin/bash
-mkdir -p ${BACKUP_DIR} && cd ${BACKUP_DIR}
+mkdir -p ${DP_BACKUP_DIR} && cd ${DP_BACKUP_DIR}
# retention 8 days by default
retention_minute=""
if [ ! -z ${LOGFILE_TTL_SECOND} ];then
retention_minute=$((${LOGFILE_TTL_SECOND}/60))
fi
export MONGODB_URI="mongodb://${DB_USER}:${DB_PASSWORD}@${DB_HOST}:27017/?authSource=admin"
-export WALG_FILE_PREFIX=${BACKUP_DIR}
+export WALG_FILE_PREFIX=${DP_BACKUP_DIR}
export OPLOG_ARCHIVE_TIMEOUT_INTERVAL=${ARCHIVE_INTERVAL}
export OPLOG_ARCHIVE_AFTER_SIZE=${ARCHIVE_AFTER_SIZE}
retryTimes=0
@@ -45,13 +45,13 @@ check_oplog_push_process(){
}
save_backup_status() {
- TOTAL_SIZE=$(du -shx ${BACKUP_DIR}|awk '{print $1}')
- OLDEST_FILE=$(ls -t ${BACKUP_DIR}/oplog_005 | tail -n 1) && OLDEST_FILE=${OLDEST_FILE#*_} && LOG_START_TIME=${OLDEST_FILE%%.*}
- LATEST_FILE=$(ls -t ${BACKUP_DIR}/oplog_005 | head -n 1) && LATEST_FILE=${LATEST_FILE##*_} && LOG_STOP_TIME=${LATEST_FILE%%.*}
+ TOTAL_SIZE=$(du -shx ${DP_BACKUP_DIR}|awk '{print $1}')
+ OLDEST_FILE=$(ls -t ${DP_BACKUP_DIR}/oplog_005 | tail -n 1) && OLDEST_FILE=${OLDEST_FILE#*_} && LOG_START_TIME=${OLDEST_FILE%%.*}
+ LATEST_FILE=$(ls -t ${DP_BACKUP_DIR}/oplog_005 | head -n 1) && LATEST_FILE=${LATEST_FILE##*_} && LOG_STOP_TIME=${LATEST_FILE%%.*}
if [ ! -z $LOG_START_TIME ]; then
START_TIME=$(date -d "@${LOG_START_TIME}" -u '+%Y-%m-%dT%H:%M:%SZ')
STOP_TIME=$(date -d "@${LOG_STOP_TIME}" -u '+%Y-%m-%dT%H:%M:%SZ')
- echo "{\"totalSize\":\"$TOTAL_SIZE\",\"manifests\":{\"backupLog\":{\"startTime\":\"${START_TIME}\",\"stopTime\":\"${STOP_TIME}\"},\"backupTool\":{\"uploadTotalSize\":\"${TOTAL_SIZE}\"}}}" > ${BACKUP_DIR}/backup.info
+ echo "{\"totalSize\":\"$TOTAL_SIZE\",\"manifests\":{\"backupLog\":{\"startTime\":\"${START_TIME}\",\"stopTime\":\"${STOP_TIME}\"},\"backupTool\":{\"uploadTotalSize\":\"${TOTAL_SIZE}\"}}}" > ${DP_BACKUP_DIR}/backup.info
fi
}
# purge the expired files
@@ -60,8 +60,8 @@ purge_expired_files() {
purgeCounter=$((purgeCounter+3))
if [ $purgeCounter -ge 60 ]; then
purgeCounter=0
- fileCount=$(find ${BACKUP_DIR}/oplog_005 -mmin +${retention_minute} -name "*.lz4" | wc -l)
- find ${BACKUP_DIR}/oplog_005 -mmin +${retention_minute} -name "*.lz4" -exec rm -rf {} \;
+ fileCount=$(find ${DP_BACKUP_DIR}/oplog_005 -mmin +${retention_minute} -name "*.lz4" | wc -l)
+ find ${DP_BACKUP_DIR}/oplog_005 -mmin +${retention_minute} -name "*.lz4" -exec rm -rf {} \;
if [ ${fileCount} -gt 0 ]; then
echo "clean up expired oplog file successfully, file count: ${fileCount}"
fi
@@ -75,7 +75,7 @@ trap "echo 'Terminating...' && kill $wal_g_pid" TERM
while true; do
check_oplog_push_process
sleep 1
- if [ -d ${BACKUP_DIR}/oplog_005 ];then
+ if [ -d ${DP_BACKUP_DIR}/oplog_005 ];then
save_backup_status
# purge the expired oplog
purge_expired_files
diff --git a/deploy/mongodb/scripts/replicaset-setup.tpl b/deploy/mongodb/scripts/replicaset-setup.tpl
index 22c4a44164e..e62c4542ddc 100644
--- a/deploy/mongodb/scripts/replicaset-setup.tpl
+++ b/deploy/mongodb/scripts/replicaset-setup.tpl
@@ -31,6 +31,7 @@ then
$CLIENT --quiet --port $PORT_FOR_RESTORE admin --eval 'db.dropUser("root", {w: "majority", wtimeout: 4000})' || true
kill $PID
wait $PID
+ echo "INFO: restore set-up configuration successfully."
rm $BACKUPFILE
fi
diff --git a/deploy/mongodb/templates/actionset-datafile.yaml b/deploy/mongodb/templates/actionset-datafile.yaml
new file mode 100644
index 00000000000..3f500d26fec
--- /dev/null
+++ b/deploy/mongodb/templates/actionset-datafile.yaml
@@ -0,0 +1,58 @@
+apiVersion: dataprotection.kubeblocks.io/v1alpha1
+kind: ActionSet
+metadata:
+ name: mongodb-physical-backup
+ labels:
+ clusterdefinition.kubeblocks.io/name: mongodb
+ {{- include "mongodb.labels" . | nindent 4 }}
+spec:
+ env:
+ - name: DATA_DIR
+ value: {{ .Values.dataMountPath }}/db
+ backupType: Full
+ backup:
+ preBackup: []
+ postBackup: []
+ backupData:
+ image: {{ .Values.image.registry | default "docker.io" }}/{{ .Values.image.repository }}:{{ .Values.image.tag }}
+ runOnTargetPodNode: true
+ syncProgress:
+ enabled: true
+ intervalSeconds: 5
+ command:
+ - bash
+ - -c
+ - |
+ {{- .Files.Get "dataprotection/backup-info-collector.sh" | nindent 8 }}
+ {{- .Files.Get "dataprotection/datafile-backup.sh" | nindent 8 }}
+
+ restore:
+ prepareData:
+ image: {{ .Values.image.registry | default "docker.io" }}/{{ .Values.image.repository }}:{{ .Values.image.tag }}
+ command:
+ - sh
+ - -c
+ - |
+ {{- .Files.Get "dataprotection/datafile-restore.sh" | nindent 8 }}
+ postReady: []
+---
+apiVersion: dataprotection.kubeblocks.io/v1alpha1
+kind: ActionSet
+metadata:
+ name: mongodb-volumesnapshot
+ labels:
+ clusterdefinition.kubeblocks.io/name: apecloud-mysql
+spec:
+ backupType: Full
+ env:
+ - name: DATA_DIR
+ value: {{ .Values.dataMountPath }}/db
+ backup: {}
+ restore:
+ prepareData:
+ image: {{ .Values.image.registry | default "docker.io" }}/{{ .Values.image.repository }}:{{ .Values.image.tag }}
+ command:
+ - sh
+ - -c
+ - "touch ${DATA_DIR}/mongodb.backup; sync"
+ postReady: []
\ No newline at end of file
diff --git a/deploy/mongodb/templates/actionset-dump.yaml b/deploy/mongodb/templates/actionset-dump.yaml
new file mode 100644
index 00000000000..97ac4a868a8
--- /dev/null
+++ b/deploy/mongodb/templates/actionset-dump.yaml
@@ -0,0 +1,39 @@
+apiVersion: dataprotection.kubeblocks.io/v1alpha1
+kind: ActionSet
+metadata:
+ name: mongodb-dump
+ labels:
+ clusterdefinition.kubeblocks.io/name: mongodb
+ {{- include "mongodb.labels" . | nindent 4 }}
+spec:
+ env:
+ - name: DATA_DIR
+ value: {{ .Values.dataMountPath }}/db
+ - name: DP_DB_PORT
+ value: "27017"
+ backupType: Full
+ backup:
+ preBackup: []
+ postBackup: []
+ backupData:
+ image: {{ .Values.image.registry | default "docker.io" }}/{{ .Values.image.repository }}:{{ .Values.image.tag }}
+ runOnTargetPodNode: false
+ syncProgress:
+ enabled: true
+ intervalSeconds: 5
+ command:
+ - bash
+ - -c
+ - |
+ {{- .Files.Get "dataprotection/backup-info-collector.sh" | nindent 8 }}
+ {{- .Files.Get "dataprotection/mongodump-backup.sh" | nindent 8 }}
+ restore:
+ postReady:
+ - job:
+ image: {{ .Values.image.registry | default "docker.io" }}/{{ .Values.image.repository }}:{{ .Values.image.tag }}
+ runOnTargetPodNode: false
+ command:
+ - sh
+ - -c
+ - |
+ {{- .Files.Get "dataprotection/mongodump-restore.sh" | nindent 10 }}
\ No newline at end of file
diff --git a/deploy/mongodb/templates/backuppolicytemplate.yaml b/deploy/mongodb/templates/backuppolicytemplate.yaml
index ced3c1835ed..fc5e359b374 100644
--- a/deploy/mongodb/templates/backuppolicytemplate.yaml
+++ b/deploy/mongodb/templates/backuppolicytemplate.yaml
@@ -9,37 +9,36 @@ spec:
clusterDefinitionRef: mongodb
backupPolicies:
- componentDefRef: mongodb
- retention:
- ttl: 7d
- schedule:
- startingDeadlineMinutes: 120
- snapshot:
- enable: false
- cronExpression: "0 18 * * *"
- datafile:
- enable: false
- cronExpression: "0 18 * * *"
- logfile:
- enable: false
- cronExpression: "*/1 * * * *"
- snapshot:
- target:
- role: primary
- hooks:
- containerName: mongodb
- preCommands:
- - "touch /data/mongodb/db/mongodb.backup; sync"
- postCommands:
- - "rm -f /data/mongodb/db/mongodb.backup; sync"
- datafile:
- backupToolName: mongodb-dump-tool
- backupsHistoryLimit: 7
- target:
- role: primary
- backupStatusUpdates:
- - updateStage: post
- useTargetPodServiceAccount: true
- logfile:
- backupToolName: mongodb-pitr-backup-tool
- target:
- role: primary
\ No newline at end of file
+ retentionPeriod: 7d
+ target:
+ role: follower
+ backupMethods:
+ - name: datafile
+ snapshotVolumes: false
+ actionSetName: mongodb-physical-backup
+ targetVolumes:
+ volumeMounts:
+ - name: data
+ mountPath: {{ .Values.dataMountPath }}
+ - name: volume-snapshot
+ snapshotVolumes: true
+ actionSetName: mongodb-volumesnapshot
+ targetVolumes:
+ volumes:
+ - data
+ volumeMounts:
+ - name: data
+ mountPath: {{ .Values.dataMountPath }}
+ - name: dump
+ snapshotVolumes: false
+ actionSetName: mongodb-dump
+ schedules:
+ - backupMethod: datafile
+ enabled: false
+ cronExpression: "0 18 * * *"
+ - backupMethod: volume-snapshot
+ enabled: false
+ cronExpression: "0 18 * * *"
+ - backupMethod: dump
+ enabled: false
+ cronExpression: "0 18 * * *"
\ No newline at end of file
diff --git a/deploy/mongodb/templates/backuptool.yaml b/deploy/mongodb/templates/backuptool.yaml
deleted file mode 100644
index 42b9a23248d..00000000000
--- a/deploy/mongodb/templates/backuptool.yaml
+++ /dev/null
@@ -1,52 +0,0 @@
-apiVersion: dataprotection.kubeblocks.io/v1alpha1
-kind: BackupTool
-metadata:
- name: mongodb-physical-backup-tool
- labels:
- clusterdefinition.kubeblocks.io/name: mongodb
- {{- include "mongodb.labels" . | nindent 4 }}
-spec:
- image: {{ .Values.image.registry | default "docker.io" }}/{{ .Values.image.repository }}:{{ .Values.image.tag }}
- deployKind: job
- env:
- - name: DATA_DIR
- value: /data/mongodb/db
- physical:
- restoreCommands:
- - sh
- - -c
- - |
- set -e
- mkdir -p ${DATA_DIR}
- res=`ls -A ${DATA_DIR}`
- data_protection_file=${DATA_DIR}/.kb-data-protection
- if [ ! -z "${res}" ] && [ ! -f ${data_protection_file} ]; then
- echo "${DATA_DIR} is not empty! Please make sure that the directory is empty before restoring the backup."
- exit 1
- fi
- touch ${data_protection_file} && sync
- tar -xvf ${BACKUP_DIR}/${BACKUP_NAME}.tar.gz -C ${DATA_DIR}
- rm -rf ${data_protection_file} && sync
- incrementalRestoreCommands: []
- logical:
- restoreCommands: []
- incrementalRestoreCommands: []
- backupCommands:
- - bash
- - -c
- - |
- if [ -d ${BACKUP_DIR} ]; then
- rm -rf ${BACKUP_DIR}
- fi
- mkdir -p ${BACKUP_DIR} && cd ${DATA_DIR}
- touch mongodb.backup && sync
- {{- .Files.Get "dataprotection/backup-info-collector.sh" | nindent 6 }}
-
- START_TIME=`get_current_time`
- # TODO: flush data and locked write, otherwise data maybe inconsistent
- tar -czvf ${BACKUP_DIR}/${BACKUP_NAME}.tar.gz ./
- rm -rf mongodb.backup
-
- # stat and save the backup information
- stat_and_save_backup_info $START_TIME
- incrementalBackupCommands: []
diff --git a/deploy/mongodb/templates/backuptool_mongodump.yaml b/deploy/mongodb/templates/backuptool_mongodump.yaml
deleted file mode 100644
index 66117013ea3..00000000000
--- a/deploy/mongodb/templates/backuptool_mongodump.yaml
+++ /dev/null
@@ -1,41 +0,0 @@
-apiVersion: dataprotection.kubeblocks.io/v1alpha1
-kind: BackupTool
-metadata:
- name: mongodb-dump-tool
- labels:
- clusterdefinition.kubeblocks.io/name: mongodb
- {{- include "mongodb.labels" . | nindent 4 }}
-spec:
- image: {{ .Values.image.registry | default "docker.io" }}/{{ .Values.image.repository }}:{{ .Values.image.tag }}
- deployKind: job
- env:
- - name: DATA_DIR
- value: /data/mongodb/db
- logical:
- restoreCommands:
- - sh
- - -c
- - |
- mongo_uri="mongodb://${DB_HOST}:27017"
- for dir_name in $(ls ${BACKUP_DIR} -l | grep ^d | awk '{print $9}'); do
- database_dir=${BACKUP_DIR}/$dir_name
- echo "INFO: restoring from ${database_dir}"
- mongorestore --uri ${mongo_uri} -u ${MONGODB_ROOT_USER} -p ${MONGODB_ROOT_PASSWORD} -d $dir_name --authenticationDatabase admin ${database_dir}
- done
- backupCommands:
- - bash
- - -c
- - |
- if [ -d ${BACKUP_DIR} ]; then
- rm -rf ${BACKUP_DIR}
- fi
- mkdir -p ${BACKUP_DIR}
- {{- .Files.Get "dataprotection/backup-info-collector.sh" | nindent 6 }}
-
- # TODO: support endpoint env for sharding cluster.
- mongo_uri="mongodb://${DB_HOST}:27017"
- START_TIME=`get_current_time`
- mongodump --uri ${mongo_uri} -u ${DB_USER} -p ${DB_PASSWORD} --authenticationDatabase admin --out ${BACKUP_DIR}
-
- # stat and save the backup information
- stat_and_save_backup_info $START_TIME
\ No newline at end of file
diff --git a/deploy/mongodb/templates/backuptool_pitr.yaml b/deploy/mongodb/templates/backuptool_pitr.yaml
deleted file mode 100644
index 04af2e45d77..00000000000
--- a/deploy/mongodb/templates/backuptool_pitr.yaml
+++ /dev/null
@@ -1,36 +0,0 @@
-apiVersion: dataprotection.kubeblocks.io/v1alpha1
-kind: BackupTool
-metadata:
- name: mongodb-pitr-backup-tool
- labels:
- clusterdefinition.kubeblocks.io/name: mongodb
- {{- include "mongodb.labels" . | nindent 4 }}
-spec:
- type: pitr
- image: {{ .Values.image.registry | default "docker.io" }}/{{ .Values.walg.repository }}:{{ .Values.walg.tag }}
- deployKind: statefulSet
- env:
- - name: ARCHIVE_AFTER_SIZE
- value: "20971520"
- physical:
- restoreCommands: []
- incrementalRestoreCommands: []
- logical:
- podScope: ReadWrite
- restoreCommands:
- - bash
- - -c
- - |
- #!/bin/bash
- set -e
- export MONGODB_URI="mongodb://${MONGODB_ROOT_USER}:${MONGODB_ROOT_PASSWORD}@${DB_HOST}:27017/?authSource=admin&replicaSet=${KB_CLUSTER_COMP_NAME}"
- export WALG_FILE_PREFIX=${BACKUP_DIR}
- echo "wal-g oplog-replay ${BASE_BACKUP_START_TIMESTAMP}.1 ${KB_RECOVERY_TIMESTAMP}.1"
- wal-g oplog-replay ${BASE_BACKUP_START_TIMESTAMP}.1 ${KB_RECOVERY_TIMESTAMP}.1
- incrementalRestoreCommands: []
- backupCommands:
- - bash
- - -c
- - |
- {{- .Files.Get "dataprotection/pitr-backup.sh" | nindent 6 }}
- incrementalBackupCommands: []
diff --git a/deploy/mongodb/templates/clusterdefinition.yaml b/deploy/mongodb/templates/clusterdefinition.yaml
index 61838f56dfe..f4e9bee0ce4 100644
--- a/deploy/mongodb/templates/clusterdefinition.yaml
+++ b/deploy/mongodb/templates/clusterdefinition.yaml
@@ -109,7 +109,7 @@ spec:
key: password
optional: false
volumeMounts:
- - mountPath: /data/mongodb
+ - mountPath: {{ .Values.dataMountPath }}
name: data
- mountPath: /etc/mongodb/mongodb.conf
name: mongodb-config
diff --git a/deploy/mongodb/values.yaml b/deploy/mongodb/values.yaml
index b0575e276a4..959e7e9d6e0 100644
--- a/deploy/mongodb/values.yaml
+++ b/deploy/mongodb/values.yaml
@@ -20,7 +20,7 @@ fullnameOverride: ""
roleProbe:
failureThreshold: 3
periodSeconds: 2
- timeoutSeconds: 1
+ timeoutSeconds: 2
## Authentication parameters
##
@@ -36,6 +36,8 @@ auth:
logConfigs:
running: /data/mongodb/logs/mongodb.log*
+dataMountPath: /data/mongodb
+
metrics:
image:
registry: registry.cn-hangzhou.aliyuncs.com
diff --git a/deploy/official-postgresql-cluster/templates/cluster.yaml b/deploy/official-postgresql-cluster/templates/cluster.yaml
index d49cc4926fa..e4b45617e62 100644
--- a/deploy/official-postgresql-cluster/templates/cluster.yaml
+++ b/deploy/official-postgresql-cluster/templates/cluster.yaml
@@ -1,15 +1,13 @@
{{- include "kblib.clusterCommon" . }}
-clusterDefinitionRef: official-postgresql
-componentSpecs:
- - name: postgresql
- componentDefRef: postgresql
+ clusterDefinitionRef: official-postgresql
+ componentSpecs:
+ - name: postgresql
+ componentDefRef: postgresql
{{- include "kblib.componentMonitor" . | indent 6 }}
{{- include "official-postgresql-cluster.replicaCount" . | indent 6 }}
- enabledLogs:
- - running
- serviceAccountName: {{ include "kblib.serviceAccountName" . }}
- switchPolicy:
- type: Noop
+ serviceAccountName: {{ include "kblib.serviceAccountName" . }}
+ switchPolicy:
+ type: Noop
{{- include "kblib.componentResources" . | indent 6 }}
{{- include "kblib.componentStorages" . | indent 6 }}
{{- include "kblib.componentServices" . | indent 6 }}
\ No newline at end of file
diff --git a/deploy/openldap-cluster/templates/cluster.yaml b/deploy/openldap-cluster/templates/cluster.yaml
index abfd9d351e4..f13b5b8b097 100644
--- a/deploy/openldap-cluster/templates/cluster.yaml
+++ b/deploy/openldap-cluster/templates/cluster.yaml
@@ -35,7 +35,7 @@ spec:
{{- end }}
{{- if .Values.persistence.enabled }}
volumeClaimTemplates:
- - name: data # ref clusterdefinition components.containers.volumeMounts.name
+ - name: openldap # ref clusterdefinition components.containers.volumeMounts.name
spec:
storageClassName: {{ .Values.persistence.data.storageClassName }}
accessModes:
diff --git a/deploy/openldap-cluster/values.yaml b/deploy/openldap-cluster/values.yaml
index 7a545409932..f0224d42356 100644
--- a/deploy/openldap-cluster/values.yaml
+++ b/deploy/openldap-cluster/values.yaml
@@ -46,7 +46,7 @@ tolerations: []
persistence:
## @param shard[*].persistence.enabled Enable persistence using Persistent Volume Claims
##
- enabled: false
+ enabled: true
## `data` volume settings
##
data:
diff --git a/deploy/openldap/templates/clusterdefinition.yaml b/deploy/openldap/templates/clusterdefinition.yaml
index 86fc0d847e8..95079821c5e 100644
--- a/deploy/openldap/templates/clusterdefinition.yaml
+++ b/deploy/openldap/templates/clusterdefinition.yaml
@@ -29,8 +29,6 @@ spec:
- mountPath: /etc/ldap/slapd.d
name: openldap
subPath: ldap-config
- - mountPath: /container/service/slapd/assets/config/bootstrap/ldif/custom
- name: openldap-bootstrap
ports:
- containerPort: 389
name: ldap
diff --git a/deploy/oracle-mysql/templates/actionset-xtrabackup.yaml b/deploy/oracle-mysql/templates/actionset-xtrabackup.yaml
new file mode 100644
index 00000000000..243bb87e2e2
--- /dev/null
+++ b/deploy/oracle-mysql/templates/actionset-xtrabackup.yaml
@@ -0,0 +1,49 @@
+apiVersion: dataprotection.kubeblocks.io/v1alpha1
+kind: ActionSet
+metadata:
+ name: oracle-mysql-xtrabackup
+ labels:
+ clusterdefinition.kubeblocks.io/name: oracle-mysql
+ {{- include "oracle-mysql.labels" . | nindent 4 }}
+spec:
+ backupType: Full
+ env:
+ - name: DATA_DIR
+ value: /var/lib/mysql
+ backup:
+ preBackup: []
+ postBackup: []
+ backupData:
+ image: docker.io/perconalab/percona-xtrabackup:8.0.32
+ runOnTargetPodNode: false
+ command:
+ - bash
+ - -c
+ - |
+ set -e;
+ mkdir -p ${DP_BACKUP_DIR};
+ xtrabackup --backup --safe-slave-backup --slave-info --stream=xbstream \
+ --host=${DP_DB_HOST} --user=${DP_DB_USER} --password=${DP_DB_PASSWORD} --datadir=${DATA_DIR} > ${DP_BACKUP_DIR}/${DP_BACKUP_NAME}.xbstream
+ TOTAL_SIZE=$(du -shx ${DP_BACKUP_DIR}|awk '{print $1}')
+ echo "{\"totalSize\":\"$TOTAL_SIZE\"}" > ${DP_BACKUP_DIR}/backup.info
+ syncProgress:
+ enabled: true
+ intervalSeconds: 5
+ restore:
+ prepareData:
+ image: docker.io/perconalab/percona-xtrabackup:8.0.32
+ command:
+ - bash
+ - -c
+ - |
+ set -e;
+ mkdir -p ${DATA_DIR}
+ TMP_DIR=/data/mysql/temp
+ mkdir -p ${TMP_DIR} && cd ${TMP_DIR}
+ xbstream -x < ${DP_BACKUP_DIR}/${DP_BACKUP_NAME}.xbstream
+ xtrabackup --decompress --remove-original --target-dir=${TMP_DIR}
+ xtrabackup --prepare --target-dir=${TMP_DIR}
+ xtrabackup --move-back --target-dir=${TMP_DIR} --datadir=${DATA_DIR}/
+ rm -rf ${TMP_DIR}
+ chmod -R 0777 ${DATA_DIR}
+ postReady: []
\ No newline at end of file
diff --git a/deploy/oracle-mysql/templates/backuppolicytemplate.yaml b/deploy/oracle-mysql/templates/backuppolicytemplate.yaml
index 2543ea7bd4b..680a6a201fe 100644
--- a/deploy/oracle-mysql/templates/backuppolicytemplate.yaml
+++ b/deploy/oracle-mysql/templates/backuppolicytemplate.yaml
@@ -9,14 +9,24 @@ spec:
clusterDefinitionRef: oracle-mysql
backupPolicies:
- componentDefRef: mysql-compdef
- schedule:
- snapshot:
- enable: true
- cronExpression: "0 18 * * *"
- datafile:
- enable: false
- cronExpression: "0 18 * * *"
- snapshot:
- backupsHistoryLimit: 5
- datafile:
- backupToolName: oracle-mysql-xtrabackup
\ No newline at end of file
+ retentionPeriod: 7d
+ backupMethods:
+ - name: xtrabackup
+ snapshotVolumes: false
+ actionSetName: oracle-mysql-xtrabackup
+ targetVolumes:
+ volumeMounts:
+ - name: data
+ mountPath: {{ .Values.dataMountPath }}
+ - name: volume-snapshot
+ snapshotVolumes: true
+ targetVolumes:
+ volumes:
+ - data
+ schedules:
+ - backupMethod: datafile
+ enabled: false
+ cronExpression: "0 18 * * 0"
+ - backupMethod: volume-snapshot
+ enabled: false
+ cronExpression: "0 18 * * 0"
\ No newline at end of file
diff --git a/deploy/oracle-mysql/templates/backuptool.yaml b/deploy/oracle-mysql/templates/backuptool.yaml
deleted file mode 100644
index 13672fcf4c8..00000000000
--- a/deploy/oracle-mysql/templates/backuptool.yaml
+++ /dev/null
@@ -1,36 +0,0 @@
-apiVersion: dataprotection.kubeblocks.io/v1alpha1
-kind: BackupTool
-metadata:
- name: oracle-mysql-xtrabackup
- labels:
- clusterdefinition.kubeblocks.io/name: oracle-mysql
- {{- include "oracle-mysql.labels" . | nindent 4 }}
-spec:
- image: docker.io/perconalab/percona-xtrabackup:8.0.32
- deployKind: job
- env:
- - name: DATA_DIR
- value: /var/lib/mysql
- physical:
- restoreCommands:
- - sh
- - -c
- - |
- set -e;
- mkdir -p ${DATA_DIR}
- TMP_DIR=/data/mysql/temp
- mkdir -p ${TMP_DIR} && cd ${TMP_DIR}
- xbstream -x < ${BACKUP_DIR}/${BACKUP_NAME}.xbstream
- xtrabackup --decompress --remove-original --target-dir=${TMP_DIR}
- xtrabackup --prepare --target-dir=${TMP_DIR}
- xtrabackup --move-back --target-dir=${TMP_DIR} --datadir=${DATA_DIR}/
- rm -rf ${TMP_DIR}
- chmod -R 0777 ${DATA_DIR}
- backupCommands:
- - sh
- - -c
- - |
- set -e;
- mkdir -p ${BACKUP_DIR};
- xtrabackup --backup --safe-slave-backup --slave-info --stream=xbstream \
- --host=${DB_HOST} --user=${DB_USER} --password=${DB_PASSWORD} --datadir=${DATA_DIR} > ${BACKUP_DIR}/${BACKUP_NAME}.xbstream
diff --git a/deploy/oracle-mysql/templates/clusterdefinition.yaml b/deploy/oracle-mysql/templates/clusterdefinition.yaml
index 927d25e7b69..24a1d016b1b 100644
--- a/deploy/oracle-mysql/templates/clusterdefinition.yaml
+++ b/deploy/oracle-mysql/templates/clusterdefinition.yaml
@@ -40,7 +40,7 @@ spec:
- name: mysql-container
imagePullPolicy: IfNotPresent
volumeMounts:
- - mountPath: /var/lib/mysql
+ - mountPath: {{ .Values.dataMountPath }}
name: data
- mountPath: /etc/mysql/conf.d
name: configs
diff --git a/deploy/oracle-mysql/values.yaml b/deploy/oracle-mysql/values.yaml
index 032a62fc03e..d960cd7c9ec 100644
--- a/deploy/oracle-mysql/values.yaml
+++ b/deploy/oracle-mysql/values.yaml
@@ -33,3 +33,5 @@ nameOverride: ""
fullnameOverride: ""
## MySQl ClusterVersion
clusterVersionOverride: "8.0.32"
+
+dataMountPath: /var/lib/mysql
diff --git a/deploy/polardbx-cluster/.helmignore b/deploy/polardbx-cluster/.helmignore
new file mode 100644
index 00000000000..a2391b0ad18
--- /dev/null
+++ b/deploy/polardbx-cluster/.helmignore
@@ -0,0 +1,24 @@
+# Patterns to ignore when building packages.
+# This supports shell glob matching, relative path matching, and
+# negation (prefixed with !). Only one pattern per line.
+.DS_Store
+# Common VCS dirs
+.git/
+.gitignore
+.bzr/
+.bzrignore
+.hg/
+.hgignore
+.svn/
+# Common backup files
+*.swp
+*.bak
+*.tmp
+*.orig
+*~
+# Various IDEs
+.project
+.idea/
+*.tmproj
+.vscode/
+*.lock
diff --git a/deploy/polardbx-cluster/Chart.yaml b/deploy/polardbx-cluster/Chart.yaml
new file mode 100644
index 00000000000..0b732a67078
--- /dev/null
+++ b/deploy/polardbx-cluster/Chart.yaml
@@ -0,0 +1,19 @@
+apiVersion: v2
+name: polardbx-cluster
+description: PolarDB-X Cluster Helm Chart for KubeBlocks.
+
+type: application
+version: 0.1.0
+appVersion: v1.4.1
+
+keywords:
+- polardbx
+- database
+- distributed
+- cloud-native
+
+home: https://polardbx.com/home
+
+maintainers:
+- name: Vettal Wu
+ email: vettal.wd@alibaba-inc.com
diff --git a/deploy/polardbx-cluster/templates/NOTES.txt b/deploy/polardbx-cluster/templates/NOTES.txt
new file mode 100644
index 00000000000..3b27cadf8f2
--- /dev/null
+++ b/deploy/polardbx-cluster/templates/NOTES.txt
@@ -0,0 +1,14 @@
+Thanks for installing PolarDB-X using KubeBlocks!
+
+1. Run the following command to create your first PolarDB-X cluster:
+
+```
+kbcli cluster create pxc --cluster-definition polardbx
+```
+
+2. Port-forward service to localhost and connect to PolarDB-X cluster:
+
+```
+kubectl port-forward svc/pxc-cn 3306:3306
+mysql -h127.0.0.1 -upolardbx_root
+```
\ No newline at end of file
diff --git a/deploy/polardbx-cluster/templates/_helpers.tpl b/deploy/polardbx-cluster/templates/_helpers.tpl
new file mode 100644
index 00000000000..d23cd372e75
--- /dev/null
+++ b/deploy/polardbx-cluster/templates/_helpers.tpl
@@ -0,0 +1,62 @@
+{{/*
+Expand the name of the chart.
+*/}}
+{{- define "polardbx.name" -}}
+{{- default .Chart.Name .Values.nameOverride | trunc 63 | trimSuffix "-" }}
+{{- end }}
+
+{{/*
+Create a default fully qualified app name.
+We truncate at 63 chars because some Kubernetes name fields are limited to this (by the DNS naming spec).
+If release name contains chart name it will be used as a full name.
+*/}}
+{{- define "polardbx.fullname" -}}
+{{- if .Values.fullnameOverride }}
+{{- .Values.fullnameOverride | trunc 63 | trimSuffix "-" }}
+{{- else }}
+{{- $name := default .Chart.Name .Values.nameOverride }}
+{{- if contains $name .Release.Name }}
+{{- .Release.Name | trunc 63 | trimSuffix "-" }}
+{{- else }}
+{{- printf "%s-%s" .Release.Name $name | trunc 63 | trimSuffix "-" }}
+{{- end }}
+{{- end }}
+{{- end }}
+
+{{/*
+Create chart name and version as used by the chart label.
+*/}}
+{{- define "polardbx.chart" -}}
+{{- printf "%s-%s" .Chart.Name .Chart.Version | replace "+" "_" | trunc 63 | trimSuffix "-" }}
+{{- end }}
+
+{{/*
+Common labels
+*/}}
+{{- define "polardbx.labels" -}}
+helm.sh/chart: {{ include "polardbx.chart" . }}
+{{ include "polardbx.selectorLabels" . }}
+{{- if .Chart.AppVersion }}
+app.kubernetes.io/version: {{ .Chart.AppVersion | quote }}
+{{- end }}
+app.kubernetes.io/managed-by: {{ .Release.Service }}
+{{- end }}
+
+{{/*
+Selector labels
+*/}}
+{{- define "polardbx.selectorLabels" -}}
+app.kubernetes.io/name: {{ include "polardbx.name" . }}
+app.kubernetes.io/instance: {{ .Release.Name }}
+{{- end }}
+
+{{/*
+Create the name of the service account to use
+*/}}
+{{- define "polardbx.serviceAccountName" -}}
+{{- if .Values.serviceAccount.create }}
+{{- default (include "polardbx.fullname" .) .Values.serviceAccount.name }}
+{{- else }}
+{{- default "default" .Values.serviceAccount.name }}
+{{- end }}
+{{- end }}
\ No newline at end of file
diff --git a/deploy/polardbx-cluster/templates/cluster.yaml b/deploy/polardbx-cluster/templates/cluster.yaml
new file mode 100644
index 00000000000..f841eef16f8
--- /dev/null
+++ b/deploy/polardbx-cluster/templates/cluster.yaml
@@ -0,0 +1,99 @@
+apiVersion: apps.kubeblocks.io/v1alpha1
+kind: Cluster
+metadata:
+ name: {{ include "polardbx.name" . }}
+ labels:
+ {{ include "polardbx.labels" . | nindent 4 }}
+spec:
+ clusterDefinitionRef: polardbx
+ clusterVersionRef: polardbx-{{ default .Chart.AppVersion .Values.clusterVersionOverride }}
+ terminationPolicy: {{ .Values.polardbx.terminationPolicy }}
+ componentSpecs:
+ - componentDefRef: gms
+ name: gms
+ replicas: {{ .Values.gms.replicas }}
+ {{- with .Values.gms.resources }}
+ resources:
+ {{- with .limits }}
+ limits:
+ cpu: {{ .cpu | quote }}
+ memory: {{ .memory | quote }}
+ {{- end }}
+ {{- with .requests }}
+ requests:
+ cpu: {{ .cpu | quote }}
+ memory: {{ .memory | quote }}
+ {{- end }}
+ {{- end }}
+ {{- if .Values.gms.persistence.enabled }}
+ volumeClaimTemplates:
+ - name: data # ref clusterdefinition components.containers.volumeMounts.name
+ spec:
+ accessModes:
+ - ReadWriteOnce
+ resources:
+ requests:
+ storage: {{ .Values.gms.persistence.data.size }}
+ {{- end }}
+ {{- $i := 0 }}
+ {{- range .Values.dn }}
+ - componentDefRef: dn
+ name: dn-{{ $i }}
+ replicas: {{ .replicas }}
+ {{- with .resources }}
+ resources:
+ {{- with .limits }}
+ limits:
+ cpu: {{ .cpu | quote }}
+ memory: {{ .memory | quote }}
+ {{- end }}
+ {{- with .requests }}
+ requests:
+ cpu: {{ .cpu | quote }}
+ memory: {{ .memory | quote }}
+ {{- end }}
+ {{- end }}
+ {{- if .persistence.enabled }}
+ volumeClaimTemplates:
+ - name: data # ref clusterdefinition components.containers.volumeMounts.name
+ spec:
+ accessModes:
+ - ReadWriteOnce
+ resources:
+ requests:
+ storage: {{ .persistence.data.size }}
+ {{- end }}
+ {{- end }}
+ {{- $i = add1 $i }}
+ - componentDefRef: cn
+ name: cn
+ replicas: {{ .Values.cn.replicas }}
+ {{- with .Values.cn.resources }}
+ resources:
+ {{- with .limits }}
+ limits:
+ cpu: {{ .cpu | quote }}
+ memory: {{ .memory | quote }}
+ {{- end }}
+ {{- with .requests }}
+ requests:
+ cpu: {{ .cpu | quote }}
+ memory: {{ .memory | quote }}
+ {{- end }}
+ {{- end }}
+ - componentDefRef: cdc
+ name: cdc
+ replicas: {{ .Values.cdc.replicas }}
+ {{- with .Values.cn.resources }}
+ resources:
+ {{- with .limits }}
+ limits:
+ cpu: {{ .cpu | quote }}
+ memory: {{ .memory | quote }}
+ {{- end }}
+ {{- with .requests }}
+ requests:
+ cpu: {{ .cpu | quote }}
+ memory: {{ .memory | quote }}
+ {{- end }}
+ {{- end }}
\ No newline at end of file
diff --git a/deploy/polardbx-cluster/values.yaml b/deploy/polardbx-cluster/values.yaml
new file mode 100644
index 00000000000..f1225cfbf9e
--- /dev/null
+++ b/deploy/polardbx-cluster/values.yaml
@@ -0,0 +1,110 @@
+## cluster settings for polardbx cluster
+nameOverride: pxc
+polardbx:
+ ## @param polardbx.terminationPolicy, temination policy for polardbx cluster
+ terminationPolicy: WipeOut
+
+gms:
+ ## @param gms.replicas data replicas of gms instance
+ ## Default value is 3, which means a paxos group: leader, follower, follower
+ replicas: 3
+
+ ## @param gms.resources
+ ## resource management for gms component
+ ## more info: https://kubernetes.io/docs/concepts/configuration/manage-resources-containers/
+ resources:
+ requests:
+ cpu: "1"
+ memory: "1Gi"
+ limits:
+ cpu: "1"
+ memory: "1Gi"
+
+
+ ## Enable persistence using Persistent Volume Claims
+ ## ref: https://kubernetes.io/docs/user-guide/persistent-volumes/
+ ##
+ persistence:
+ ## @param shard[*].persistence.enabled Enable persistence using Persistent Volume Claims
+ ##
+ enabled: true
+ ## `data` volume settings
+ ##
+ data:
+ ## @param shard[*].persistence.data.storageClassName Storage class of backing PVC
+ ## If defined, storageClassName:
+ ## If set to "-", storageClassName: "", which disables dynamic provisioning
+ ## If undefined (the default) or set to null, no storageClassName spec is
+ ## set, choosing the default provisioner. (gp2 on AWS, standard on
+ ## GKE, AWS & OpenStack)
+ ##
+ storageClassName:
+ ## @param shard[*].persistence.size Size of data volume
+ ##
+ size: 20Gi
+
+dn:
+ -
+ ## @param dn[*].replicas data replicas of each DN instance
+ ## Default value is 3, which means a paxos group: leader, follower, follower
+ replicas: 3
+ ## @param dn[*].resources
+ ## resource management for dn component
+ ## more info: https://kubernetes.io/docs/concepts/configuration/manage-resources-containers/
+ resources:
+ requests:
+ cpu: "1"
+ memory: "1Gi"
+ limits:
+ cpu: "1"
+ memory: "1Gi"
+
+ ## Enable persistence using Persistent Volume Claims
+ ## ref: https://kubernetes.io/docs/user-guide/persistent-volumes/
+ ##
+ persistence:
+ ## @param shard[*].persistence.enabled Enable persistence using Persistent Volume Claims
+ ##
+ enabled: true
+ ## `data` volume settings
+ ##
+ data:
+ ## @param shard[*].persistence.data.storageClassName Storage class of backing PVC
+ ## If defined, storageClassName:
+ ## If set to "-", storageClassName: "", which disables dynamic provisioning
+ ## If undefined (the default) or set to null, no storageClassName spec is
+ ## set, choosing the default provisioner. (gp2 on AWS, standard on
+ ## GKE, AWS & OpenStack)
+ ##
+ storageClassName:
+ ## @param shard[*].persistence.size Size of data volume
+ ##
+ size: 20Gi
+
+cn:
+ ## @param cn.replicas number of polardb-x cn nodes
+ replicas: 2
+ ## @param cn.resources
+ ## resource management for cn component
+ ## more info: https://kubernetes.io/docs/concepts/configuration/manage-resources-containers/
+ resources:
+ requests:
+ cpu: "1"
+ memory: "1Gi"
+ limits:
+ cpu: "1"
+ memory: "1Gi"
+
+cdc:
+ ## @param cdc.replicas number of polardb-x cdc nodes
+ replicas: 2
+ ## @param cdc.resources
+ ## resource management for cdc component
+ ## more info: https://kubernetes.io/docs/concepts/configuration/manage-resources-containers/
+ resources:
+ requests:
+ cpu: "1"
+ memory: "1Gi"
+ limits:
+ cpu: "1"
+ memory: "1Gi"
\ No newline at end of file
diff --git a/deploy/polardbx/.helmignore b/deploy/polardbx/.helmignore
new file mode 100644
index 00000000000..0e8a0eb36f4
--- /dev/null
+++ b/deploy/polardbx/.helmignore
@@ -0,0 +1,23 @@
+# Patterns to ignore when building packages.
+# This supports shell glob matching, relative path matching, and
+# negation (prefixed with !). Only one pattern per line.
+.DS_Store
+# Common VCS dirs
+.git/
+.gitignore
+.bzr/
+.bzrignore
+.hg/
+.hgignore
+.svn/
+# Common backup files
+*.swp
+*.bak
+*.tmp
+*.orig
+*~
+# Various IDEs
+.project
+.idea/
+*.tmproj
+.vscode/
diff --git a/deploy/polardbx/Chart.yaml b/deploy/polardbx/Chart.yaml
new file mode 100644
index 00000000000..e78eb94f5a5
--- /dev/null
+++ b/deploy/polardbx/Chart.yaml
@@ -0,0 +1,19 @@
+apiVersion: v2
+name: polardbx
+description: PolarDB-X Cluster Helm Chart for KubeBlocks.
+
+type: application
+version: 0.1.0
+appVersion: v1.4.1
+
+keywords:
+ - polardbx
+ - database
+ - distributed
+ - cloud-native
+
+home: https://polardbx.com/home
+
+maintainers:
+ - name: Vettal Wu
+ email: vettal.wd@alibaba-inc.com
\ No newline at end of file
diff --git a/deploy/polardbx/dashboards/polardbx-overview.json b/deploy/polardbx/dashboards/polardbx-overview.json
new file mode 100644
index 00000000000..fac3adf8099
--- /dev/null
+++ b/deploy/polardbx/dashboards/polardbx-overview.json
@@ -0,0 +1,4626 @@
+{
+ "annotations": {
+ "list": [
+ {
+ "builtIn": 1,
+ "datasource": {
+ "type": "datasource",
+ "uid": "grafana"
+ },
+ "enable": true,
+ "hide": true,
+ "iconColor": "rgba(0, 211, 255, 1)",
+ "name": "Annotations & Alerts",
+ "target": {
+ "limit": 100,
+ "matchAny": false,
+ "tags": [],
+ "type": "dashboard"
+ },
+ "type": "dashboard"
+ }
+ ]
+ },
+ "description": "",
+ "editable": true,
+ "fiscalYearStartMonth": 0,
+ "graphTooltip": 0,
+ "links": [],
+ "liveNow": false,
+ "panels": [
+ {
+ "collapsed": false,
+ "datasource": {
+ "type": "prometheus",
+ "uid": "prometheus"
+ },
+ "gridPos": {
+ "h": 1,
+ "w": 24,
+ "x": 0,
+ "y": 0
+ },
+ "id": 2,
+ "panels": [],
+ "targets": [
+ {
+ "datasource": {
+ "type": "prometheus",
+ "uid": "prometheus"
+ },
+ "refId": "A"
+ }
+ ],
+ "title": "Overview",
+ "type": "row"
+ },
+ {
+ "datasource": {
+ "type": "prometheus",
+ "uid": "${datasource}"
+ },
+ "fieldConfig": {
+ "defaults": {
+ "color": {
+ "mode": "palette-classic"
+ },
+ "mappings": [],
+ "thresholds": {
+ "mode": "percentage",
+ "steps": [
+ {
+ "color": "green",
+ "value": null
+ }
+ ]
+ },
+ "unit": "none"
+ },
+ "overrides": []
+ },
+ "gridPos": {
+ "h": 4,
+ "w": 7,
+ "x": 0,
+ "y": 1
+ },
+ "id": 15,
+ "options": {
+ "colorMode": "value",
+ "graphMode": "none",
+ "justifyMode": "auto",
+ "orientation": "auto",
+ "reduceOptions": {
+ "calcs": [
+ "mean"
+ ],
+ "fields": "",
+ "values": false
+ },
+ "text": {},
+ "textMode": "auto"
+ },
+ "pluginVersion": "9.2.4",
+ "targets": [
+ {
+ "datasource": {
+ "uid": "${datasource}"
+ },
+ "editorMode": "code",
+ "exemplar": true,
+ "expr": "sum(irate(polardbx_stats_request_count_total{app_kubernetes_io_instance=\"$polardbx\", namespace=\"$namespace\"}[1m]))",
+ "instant": true,
+ "interval": "",
+ "legendFormat": "Logical",
+ "queryType": "randomWalk",
+ "refId": "A"
+ },
+ {
+ "datasource": {
+ "uid": "${datasource}"
+ },
+ "editorMode": "code",
+ "exemplar": true,
+ "expr": "sum(irate(polardbx_stats_physical_request_count_total{app_kubernetes_io_instance=\"$polardbx\", namespace=\"$namespace\"}[1m]))",
+ "hide": true,
+ "instant": true,
+ "interval": "",
+ "legendFormat": "Physical",
+ "refId": "B"
+ }
+ ],
+ "title": "QPS (Logical)",
+ "type": "stat"
+ },
+ {
+ "datasource": {
+ "type": "prometheus",
+ "uid": "${datasource}"
+ },
+ "fieldConfig": {
+ "defaults": {
+ "color": {
+ "mode": "palette-classic"
+ },
+ "mappings": [],
+ "min": 0,
+ "thresholds": {
+ "mode": "absolute",
+ "steps": [
+ {
+ "color": "green",
+ "value": null
+ }
+ ]
+ },
+ "unit": "none"
+ },
+ "overrides": []
+ },
+ "gridPos": {
+ "h": 4,
+ "w": 7,
+ "x": 7,
+ "y": 1
+ },
+ "id": 10,
+ "options": {
+ "displayMode": "gradient",
+ "minVizHeight": 10,
+ "minVizWidth": 0,
+ "orientation": "horizontal",
+ "reduceOptions": {
+ "calcs": [
+ "mean"
+ ],
+ "fields": "",
+ "values": false
+ },
+ "showUnfilled": true,
+ "text": {}
+ },
+ "pluginVersion": "9.2.4",
+ "targets": [
+ {
+ "datasource": {
+ "uid": "${datasource}"
+ },
+ "editorMode": "code",
+ "exemplar": true,
+ "expr": "sum(irate(polardbx_stats_request_count_total{app_kubernetes_io_instance=\"$polardbx\", namespace=\"$namespace\"}[1m]))",
+ "instant": true,
+ "interval": "",
+ "legendFormat": "QPS (Logical)",
+ "queryType": "randomWalk",
+ "refId": "A"
+ },
+ {
+ "datasource": {
+ "uid": "${datasource}"
+ },
+ "editorMode": "code",
+ "exemplar": true,
+ "expr": "sum(irate(polardbx_stats_physical_request_count_total{app_kubernetes_io_instance=\"$polardbx\", namespace=\"$namespace\"}[1m]))",
+ "hide": false,
+ "instant": true,
+ "interval": "",
+ "legendFormat": "QPS (Physical)",
+ "refId": "B"
+ }
+ ],
+ "title": "QPS",
+ "type": "bargauge"
+ },
+ {
+ "datasource": {
+ "type": "prometheus",
+ "uid": "${datasource}"
+ },
+ "fieldConfig": {
+ "defaults": {
+ "color": {
+ "mode": "palette-classic"
+ },
+ "mappings": [],
+ "thresholds": {
+ "mode": "absolute",
+ "steps": [
+ {
+ "color": "green",
+ "value": null
+ }
+ ]
+ }
+ },
+ "overrides": []
+ },
+ "gridPos": {
+ "h": 4,
+ "w": 5,
+ "x": 14,
+ "y": 1
+ },
+ "id": 40,
+ "options": {
+ "colorMode": "value",
+ "graphMode": "none",
+ "justifyMode": "auto",
+ "orientation": "auto",
+ "reduceOptions": {
+ "calcs": [
+ "lastNotNull"
+ ],
+ "fields": "",
+ "values": false
+ },
+ "text": {},
+ "textMode": "auto"
+ },
+ "pluginVersion": "9.2.4",
+ "targets": [
+ {
+ "datasource": {
+ "uid": "${datasource}"
+ },
+ "editorMode": "code",
+ "exemplar": true,
+ "expr": "sum(rate(polardbx_stats_best_effort_transaction_count_total{app_kubernetes_io_instance=\"$polardbx\", namespace=\"$namespace\"}[1m])) +\n\nsum(rate(polardbx_stats_xa_transaction_count_total{app_kubernetes_io_instance=\"$polardbx\", namespace=\"$namespace\"}[1m])) +\n\nsum(rate(polardbx_stats_tso_transaction_count_total{app_kubernetes_io_instance=\"$polardbx\", namespace=\"$namespace\"}[1m])) ",
+ "instant": true,
+ "interval": "",
+ "intervalFactor": 1,
+ "legendFormat": "Errors",
+ "queryType": "randomWalk",
+ "refId": "A"
+ }
+ ],
+ "title": "TPS",
+ "type": "stat"
+ },
+ {
+ "datasource": {
+ "type": "prometheus",
+ "uid": "${datasource}"
+ },
+ "fieldConfig": {
+ "defaults": {
+ "color": {
+ "mode": "palette-classic"
+ },
+ "mappings": [],
+ "min": 0,
+ "thresholds": {
+ "mode": "absolute",
+ "steps": [
+ {
+ "color": "green",
+ "value": null
+ },
+ {
+ "color": "red",
+ "value": 80
+ }
+ ]
+ }
+ },
+ "overrides": []
+ },
+ "gridPos": {
+ "h": 4,
+ "w": 5,
+ "x": 19,
+ "y": 1
+ },
+ "id": 14,
+ "options": {
+ "displayMode": "gradient",
+ "minVizHeight": 10,
+ "minVizWidth": 0,
+ "orientation": "horizontal",
+ "reduceOptions": {
+ "calcs": [
+ "lastNotNull"
+ ],
+ "fields": "",
+ "values": false
+ },
+ "showUnfilled": true,
+ "text": {}
+ },
+ "pluginVersion": "9.2.4",
+ "targets": [
+ {
+ "datasource": {
+ "uid": "${datasource}"
+ },
+ "editorMode": "code",
+ "exemplar": true,
+ "expr": "sum(polardbx_stats_active_connections{app_kubernetes_io_instance=\"$polardbx\", namespace=\"$namespace\"})",
+ "instant": true,
+ "interval": "",
+ "legendFormat": "Connections",
+ "queryType": "randomWalk",
+ "refId": "A"
+ },
+ {
+ "datasource": {
+ "uid": "${datasource}"
+ },
+ "editorMode": "code",
+ "exemplar": true,
+ "expr": "sum(polardbx_stats_running_count{app_kubernetes_io_instance=\"$polardbx\", namespace=\"$namespace\"})",
+ "hide": false,
+ "instant": true,
+ "interval": "",
+ "legendFormat": "Threads (Running)",
+ "refId": "B"
+ }
+ ],
+ "title": "Connections/Threads",
+ "type": "bargauge"
+ },
+ {
+ "datasource": {
+ "type": "prometheus",
+ "uid": "${datasource}"
+ },
+ "fieldConfig": {
+ "defaults": {
+ "color": {
+ "mode": "thresholds"
+ },
+ "mappings": [],
+ "thresholds": {
+ "mode": "absolute",
+ "steps": [
+ {
+ "color": "green",
+ "value": null
+ },
+ {
+ "color": "dark-yellow",
+ "value": 20
+ },
+ {
+ "color": "dark-orange",
+ "value": 50
+ },
+ {
+ "color": "dark-red",
+ "value": 100
+ }
+ ]
+ },
+ "unit": "ms"
+ },
+ "overrides": []
+ },
+ "gridPos": {
+ "h": 4,
+ "w": 7,
+ "x": 0,
+ "y": 5
+ },
+ "id": 16,
+ "options": {
+ "colorMode": "value",
+ "graphMode": "none",
+ "justifyMode": "auto",
+ "orientation": "auto",
+ "reduceOptions": {
+ "calcs": [
+ "mean"
+ ],
+ "fields": "",
+ "values": false
+ },
+ "text": {},
+ "textMode": "auto"
+ },
+ "pluginVersion": "9.2.4",
+ "targets": [
+ {
+ "datasource": {
+ "uid": "${datasource}"
+ },
+ "editorMode": "code",
+ "exemplar": true,
+ "expr": "sum(delta(polardbx_stats_request_time_cost_total{app_kubernetes_io_instance=\"$polardbx\", namespace=\"$namespace\"}[1m])) / (sum(delta (polardbx_stats_request_count_total{app_kubernetes_io_instance=\"$polardbx\", namespace=\"$namespace\"}[1m])) + 1) / 1000",
+ "instant": true,
+ "interval": "",
+ "legendFormat": "Logical",
+ "queryType": "randomWalk",
+ "refId": "A"
+ }
+ ],
+ "title": "Response Time (Logical)",
+ "type": "stat"
+ },
+ {
+ "datasource": {
+ "type": "prometheus",
+ "uid": "${datasource}"
+ },
+ "fieldConfig": {
+ "defaults": {
+ "color": {
+ "mode": "thresholds"
+ },
+ "decimals": 2,
+ "mappings": [],
+ "min": 0,
+ "thresholds": {
+ "mode": "absolute",
+ "steps": [
+ {
+ "color": "green",
+ "value": null
+ },
+ {
+ "color": "light-yellow",
+ "value": 20
+ },
+ {
+ "color": "dark-orange",
+ "value": 50
+ },
+ {
+ "color": "red",
+ "value": 100
+ }
+ ]
+ },
+ "unit": "ms"
+ },
+ "overrides": []
+ },
+ "gridPos": {
+ "h": 4,
+ "w": 7,
+ "x": 7,
+ "y": 5
+ },
+ "id": 12,
+ "options": {
+ "displayMode": "gradient",
+ "minVizHeight": 10,
+ "minVizWidth": 0,
+ "orientation": "horizontal",
+ "reduceOptions": {
+ "calcs": [
+ "lastNotNull"
+ ],
+ "fields": "",
+ "values": false
+ },
+ "showUnfilled": true,
+ "text": {}
+ },
+ "pluginVersion": "9.2.4",
+ "targets": [
+ {
+ "datasource": {
+ "uid": "${datasource}"
+ },
+ "editorMode": "code",
+ "exemplar": true,
+ "expr": "sum(delta(polardbx_stats_request_time_cost_total{app_kubernetes_io_instance=\"$polardbx\", namespace=\"$namespace\"}[1m])) / (sum(delta (polardbx_stats_request_count_total{app_kubernetes_io_instance=\"$polardbx\", namespace=\"$namespace\"}[1m])) + 1) / 1000",
+ "instant": true,
+ "interval": "",
+ "legendFormat": "RT (Logical)",
+ "queryType": "randomWalk",
+ "refId": "A"
+ },
+ {
+ "datasource": {
+ "uid": "${datasource}"
+ },
+ "editorMode": "code",
+ "exemplar": true,
+ "expr": "sum(delta(polardbx_stats_physical_request_time_cost_total{app_kubernetes_io_instance=\"$polardbx\", namespace=\"$namespace\"}[1m])) / (sum(delta (polardbx_stats_physical_request_count_total{app_kubernetes_io_instance=\"$polardbx\", namespace=\"$namespace\"}[1m])) + 1) / 1000",
+ "hide": false,
+ "instant": true,
+ "interval": "",
+ "legendFormat": "RT (Physical)",
+ "refId": "B"
+ }
+ ],
+ "title": "Response Time",
+ "type": "bargauge"
+ },
+ {
+ "datasource": {
+ "type": "prometheus",
+ "uid": "${datasource}"
+ },
+ "fieldConfig": {
+ "defaults": {
+ "color": {
+ "mode": "thresholds"
+ },
+ "mappings": [],
+ "thresholds": {
+ "mode": "absolute",
+ "steps": [
+ {
+ "color": "green",
+ "value": null
+ },
+ {
+ "color": "#EAB839",
+ "value": 15
+ },
+ {
+ "color": "red",
+ "value": 30
+ }
+ ]
+ },
+ "unit": "none"
+ },
+ "overrides": []
+ },
+ "gridPos": {
+ "h": 4,
+ "w": 5,
+ "x": 14,
+ "y": 5
+ },
+ "id": 18,
+ "options": {
+ "colorMode": "value",
+ "graphMode": "none",
+ "justifyMode": "auto",
+ "orientation": "auto",
+ "reduceOptions": {
+ "calcs": [
+ "lastNotNull"
+ ],
+ "fields": "",
+ "values": false
+ },
+ "text": {},
+ "textMode": "auto"
+ },
+ "pluginVersion": "9.2.4",
+ "targets": [
+ {
+ "datasource": {
+ "uid": "${datasource}"
+ },
+ "editorMode": "code",
+ "exemplar": true,
+ "expr": "sum(rate(polardbx_stats_error_count_total{app_kubernetes_io_instance=\"$polardbx\", namespace=\"$namespace\"}[1m]))",
+ "instant": true,
+ "interval": "",
+ "intervalFactor": 1,
+ "legendFormat": "Errors",
+ "queryType": "randomWalk",
+ "refId": "A"
+ }
+ ],
+ "title": "Errors",
+ "type": "stat"
+ },
+ {
+ "datasource": {
+ "type": "prometheus",
+ "uid": "${datasource}"
+ },
+ "fieldConfig": {
+ "defaults": {
+ "color": {
+ "mode": "palette-classic"
+ },
+ "custom": {
+ "align": "center",
+ "displayMode": "auto",
+ "filterable": false,
+ "inspect": false
+ },
+ "mappings": [],
+ "thresholds": {
+ "mode": "absolute",
+ "steps": [
+ {
+ "color": "green",
+ "value": null
+ }
+ ]
+ }
+ },
+ "overrides": [
+ {
+ "matcher": {
+ "id": "byName",
+ "options": "CPU %"
+ },
+ "properties": [
+ {
+ "id": "unit",
+ "value": "percentunit"
+ },
+ {
+ "id": "custom.displayMode",
+ "value": "gradient-gauge"
+ }
+ ]
+ },
+ {
+ "matcher": {
+ "id": "byName",
+ "options": "MEM %"
+ },
+ "properties": [
+ {
+ "id": "unit",
+ "value": "percentunit"
+ },
+ {
+ "id": "custom.displayMode",
+ "value": "gradient-gauge"
+ }
+ ]
+ },
+ {
+ "matcher": {
+ "id": "byName",
+ "options": "MEM"
+ },
+ "properties": [
+ {
+ "id": "unit",
+ "value": "bytes"
+ },
+ {
+ "id": "custom.width",
+ "value": 162
+ }
+ ]
+ },
+ {
+ "matcher": {
+ "id": "byName",
+ "options": "MEM (Limit)"
+ },
+ "properties": [
+ {
+ "id": "unit",
+ "value": "bytes"
+ }
+ ]
+ },
+ {
+ "matcher": {
+ "id": "byName",
+ "options": "Pod"
+ },
+ "properties": [
+ {
+ "id": "custom.filterable",
+ "value": true
+ },
+ {
+ "id": "custom.width",
+ "value": 215
+ }
+ ]
+ },
+ {
+ "matcher": {
+ "id": "byName",
+ "options": "Network (Recv)"
+ },
+ "properties": [
+ {
+ "id": "unit",
+ "value": "bytes"
+ },
+ {
+ "id": "custom.displayMode",
+ "value": "auto"
+ }
+ ]
+ },
+ {
+ "matcher": {
+ "id": "byName",
+ "options": "Network (Sent)"
+ },
+ "properties": [
+ {
+ "id": "unit",
+ "value": "bytes"
+ }
+ ]
+ },
+ {
+ "matcher": {
+ "id": "byName",
+ "options": "node"
+ },
+ "properties": [
+ {
+ "id": "custom.width",
+ "value": 205
+ }
+ ]
+ },
+ {
+ "matcher": {
+ "id": "byName",
+ "options": "Host"
+ },
+ "properties": [
+ {
+ "id": "links",
+ "value": [
+ {
+ "targetBlank": true,
+ "title": "Node Dashboard",
+ "url": "./d/fa49a4706d07a042595b664c87fb33ea/nodes?orgId=1&var-datasource=$datasource&var-instance=${__value.text}"
+ }
+ ]
+ },
+ {
+ "id": "custom.width",
+ "value": 221
+ }
+ ]
+ },
+ {
+ "matcher": {
+ "id": "byName",
+ "options": "Pod"
+ },
+ "properties": [
+ {
+ "id": "links",
+ "value": [
+ {
+ "targetBlank": true,
+ "title": "Pod Dashboard",
+ "url": "./d/6581e46e4e5c7ba40a07646395ef7b23/k8s-resources-pod?var-datasource=$datasource&var-cluster=$cluster&var-namespace=$namespace&var-pod=${__value.text}"
+ }
+ ]
+ }
+ ]
+ },
+ {
+ "matcher": {
+ "id": "byName",
+ "options": "CPU"
+ },
+ "properties": [
+ {
+ "id": "custom.width",
+ "value": 92
+ }
+ ]
+ },
+ {
+ "matcher": {
+ "id": "byName",
+ "options": "CPU (Limit)"
+ },
+ "properties": [
+ {
+ "id": "custom.width",
+ "value": 107
+ }
+ ]
+ }
+ ]
+ },
+ "gridPos": {
+ "h": 10,
+ "w": 24,
+ "x": 0,
+ "y": 9
+ },
+ "id": 96,
+ "options": {
+ "footer": {
+ "fields": "",
+ "reducer": [
+ "sum"
+ ],
+ "show": false
+ },
+ "frameIndex": 0,
+ "showHeader": true,
+ "sortBy": [
+ {
+ "desc": false,
+ "displayName": "CPU"
+ }
+ ]
+ },
+ "pluginVersion": "9.2.4",
+ "targets": [
+ {
+ "datasource": {
+ "uid": "${datasource}"
+ },
+ "editorMode": "code",
+ "exemplar": true,
+ "expr": "sum by (app_kubernetes_io_instance, pod, app_kubernetes_io_component, apps_kubeblocks_io_component_name) (mysql_up{app_kubernetes_io_instance=\"$polardbx\", namespace=\"$namespace\", app_kubernetes_io_component=\"gms\"})\nor\nsum by (app_kubernetes_io_instance, pod, app_kubernetes_io_component, apps_kubeblocks_io_component_name) (polardbx_up{app_kubernetes_io_instance=\"$polardbx\", namespace=\"$namespace\"})\nor \nsum by (app_kubernetes_io_instance, pod, app_kubernetes_io_component, apps_kubeblocks_io_component_name) (mysql_up{app_kubernetes_io_instance=\"$polardbx\", namespace=\"$namespace\", app_kubernetes_io_component=\"dn\"})\nor \nsum by (app_kubernetes_io_instance, pod, app_kubernetes_io_component, apps_kubeblocks_io_component_name) (polardbx_cdc_up{app_kubernetes_io_instance=\"$polardbx\", namespace=\"$namespace\"})\n",
+ "format": "table",
+ "hide": false,
+ "instant": true,
+ "interval": "",
+ "legendFormat": "{{ pod }}",
+ "queryType": "randomWalk",
+ "refId": "A"
+ }
+ ],
+ "title": "Topology",
+ "transformations": [
+ {
+ "id": "seriesToColumns",
+ "options": {
+ "byField": "pod"
+ }
+ },
+ {
+ "id": "organize",
+ "options": {
+ "excludeByName": {
+ "Time": true,
+ "Time 1": true,
+ "Time 2": true,
+ "Time 3": true,
+ "Time 4": true,
+ "Time 5": true,
+ "Time 6": true,
+ "Time 7": true,
+ "Time 8": true,
+ "Time 9": true,
+ "Value": true,
+ "Value #A": true
+ },
+ "indexByName": {
+ "Time": 2,
+ "Value": 5,
+ "app_kubernetes_io_component": 1,
+ "app_kubernetes_io_instance": 0,
+ "apps_kubeblocks_io_component_name": 3,
+ "pod": 4
+ },
+ "renameByName": {
+ "Value #B": "CPU",
+ "Value #C": "CPU (Limit)",
+ "Value #D": "CPU %",
+ "Value #E": "MEM",
+ "Value #F": "MEM (Limit)",
+ "Value #G": "MEM %",
+ "Value #H": "Network (Recv)",
+ "Value #I": "Network (Sent)",
+ "app_kubernetes_io_component": "role",
+ "app_kubernetes_io_instance": "instance",
+ "apps_kubeblocks_io_component_name": "component_name",
+ "node": "Host",
+ "pod": "pod"
+ }
+ }
+ },
+ {
+ "id": "sortBy",
+ "options": {
+ "fields": {},
+ "sort": [
+ {
+ "field": "Pod"
+ }
+ ]
+ }
+ }
+ ],
+ "type": "table"
+ },
+ {
+ "collapsed": false,
+ "datasource": {
+ "type": "prometheus",
+ "uid": "prometheus"
+ },
+ "gridPos": {
+ "h": 1,
+ "w": 24,
+ "x": 0,
+ "y": 19
+ },
+ "id": 42,
+ "panels": [],
+ "targets": [
+ {
+ "datasource": {
+ "type": "prometheus",
+ "uid": "prometheus"
+ },
+ "refId": "A"
+ }
+ ],
+ "title": "Global Meta Service",
+ "type": "row"
+ },
+ {
+ "datasource": {
+ "type": "prometheus",
+ "uid": "${datasource}"
+ },
+ "fieldConfig": {
+ "defaults": {
+ "color": {
+ "mode": "palette-classic"
+ },
+ "custom": {
+ "axisCenteredZero": false,
+ "axisColorMode": "text",
+ "axisLabel": "",
+ "axisPlacement": "auto",
+ "barAlignment": 0,
+ "drawStyle": "line",
+ "fillOpacity": 10,
+ "gradientMode": "opacity",
+ "hideFrom": {
+ "graph": false,
+ "legend": false,
+ "tooltip": false,
+ "viz": false
+ },
+ "lineInterpolation": "linear",
+ "lineStyle": {
+ "fill": "solid"
+ },
+ "lineWidth": 1,
+ "pointSize": 5,
+ "scaleDistribution": {
+ "type": "linear"
+ },
+ "showPoints": "never",
+ "spanNulls": true,
+ "stacking": {
+ "group": "A",
+ "mode": "none"
+ },
+ "thresholdsStyle": {
+ "mode": "off"
+ }
+ },
+ "mappings": [],
+ "thresholds": {
+ "mode": "absolute",
+ "steps": [
+ {
+ "color": "green",
+ "value": null
+ }
+ ]
+ },
+ "unit": "short"
+ },
+ "overrides": []
+ },
+ "gridPos": {
+ "h": 8,
+ "w": 6,
+ "x": 0,
+ "y": 20
+ },
+ "id": 86,
+ "options": {
+ "graph": {},
+ "legend": {
+ "calcs": [],
+ "displayMode": "list",
+ "placement": "bottom",
+ "showLegend": true
+ },
+ "tooltip": {
+ "mode": "single",
+ "sort": "none"
+ }
+ },
+ "pluginVersion": "7.5.3",
+ "targets": [
+ {
+ "datasource": {
+ "uid": "${datasource}"
+ },
+ "editorMode": "code",
+ "exemplar": true,
+ "expr": "sum(rate(mysql_global_status_queries{app_kubernetes_io_instance=\"$polardbx\", app_kubernetes_io_component=\"gms\", namespace=\"$namespace\"}[1m]))",
+ "interval": "",
+ "legendFormat": "qps (physical)",
+ "queryType": "randomWalk",
+ "range": true,
+ "refId": "A"
+ }
+ ],
+ "title": "QPS",
+ "type": "timeseries"
+ },
+ {
+ "datasource": {
+ "type": "prometheus",
+ "uid": "${datasource}"
+ },
+ "fieldConfig": {
+ "defaults": {
+ "color": {
+ "mode": "palette-classic"
+ },
+ "custom": {
+ "axisCenteredZero": false,
+ "axisColorMode": "text",
+ "axisLabel": "",
+ "axisPlacement": "auto",
+ "barAlignment": 0,
+ "drawStyle": "line",
+ "fillOpacity": 10,
+ "gradientMode": "opacity",
+ "hideFrom": {
+ "graph": false,
+ "legend": false,
+ "tooltip": false,
+ "viz": false
+ },
+ "lineInterpolation": "linear",
+ "lineStyle": {
+ "fill": "solid"
+ },
+ "lineWidth": 1,
+ "pointSize": 5,
+ "scaleDistribution": {
+ "type": "linear"
+ },
+ "showPoints": "never",
+ "spanNulls": true,
+ "stacking": {
+ "group": "A",
+ "mode": "none"
+ },
+ "thresholdsStyle": {
+ "mode": "off"
+ }
+ },
+ "mappings": [],
+ "thresholds": {
+ "mode": "absolute",
+ "steps": [
+ {
+ "color": "green",
+ "value": null
+ }
+ ]
+ },
+ "unit": "short"
+ },
+ "overrides": []
+ },
+ "gridPos": {
+ "h": 8,
+ "w": 6,
+ "x": 6,
+ "y": 20
+ },
+ "id": 87,
+ "options": {
+ "graph": {},
+ "legend": {
+ "calcs": [],
+ "displayMode": "list",
+ "placement": "bottom",
+ "showLegend": true
+ },
+ "tooltip": {
+ "mode": "multi",
+ "sort": "none"
+ }
+ },
+ "pluginVersion": "7.5.3",
+ "targets": [
+ {
+ "datasource": {
+ "uid": "${datasource}"
+ },
+ "editorMode": "code",
+ "exemplar": true,
+ "expr": "sum(mysql_global_status_threads_connected{app_kubernetes_io_instance=\"$polardbx\", app_kubernetes_io_component=\"gms\", namespace=\"$namespace\"})",
+ "interval": "",
+ "legendFormat": "connections",
+ "queryType": "randomWalk",
+ "range": true,
+ "refId": "A"
+ },
+ {
+ "datasource": {
+ "uid": "${datasource}"
+ },
+ "editorMode": "code",
+ "exemplar": true,
+ "expr": "sum(mysql_global_status_threads_running{app_kubernetes_io_instance=\"$polardbx\", app_kubernetes_io_component=\"gms\", namespace=\"$namespace\"})",
+ "hide": false,
+ "interval": "",
+ "legendFormat": "threads",
+ "range": true,
+ "refId": "B"
+ }
+ ],
+ "title": "Connections/Threads",
+ "type": "timeseries"
+ },
+ {
+ "datasource": {
+ "type": "prometheus",
+ "uid": "${datasource}"
+ },
+ "fieldConfig": {
+ "defaults": {
+ "color": {
+ "mode": "palette-classic"
+ },
+ "custom": {
+ "axisCenteredZero": false,
+ "axisColorMode": "text",
+ "axisLabel": "",
+ "axisPlacement": "auto",
+ "barAlignment": 0,
+ "drawStyle": "line",
+ "fillOpacity": 10,
+ "gradientMode": "opacity",
+ "hideFrom": {
+ "graph": false,
+ "legend": false,
+ "tooltip": false,
+ "viz": false
+ },
+ "lineInterpolation": "linear",
+ "lineStyle": {
+ "fill": "solid"
+ },
+ "lineWidth": 1,
+ "pointSize": 5,
+ "scaleDistribution": {
+ "type": "linear"
+ },
+ "showPoints": "never",
+ "spanNulls": true,
+ "stacking": {
+ "group": "A",
+ "mode": "none"
+ },
+ "thresholdsStyle": {
+ "mode": "off"
+ }
+ },
+ "mappings": [],
+ "thresholds": {
+ "mode": "absolute",
+ "steps": [
+ {
+ "color": "green",
+ "value": null
+ }
+ ]
+ },
+ "unit": "bytes"
+ },
+ "overrides": []
+ },
+ "gridPos": {
+ "h": 8,
+ "w": 6,
+ "x": 12,
+ "y": 20
+ },
+ "id": 88,
+ "options": {
+ "graph": {},
+ "legend": {
+ "calcs": [],
+ "displayMode": "list",
+ "placement": "bottom",
+ "showLegend": true
+ },
+ "tooltip": {
+ "mode": "single",
+ "sort": "none"
+ }
+ },
+ "pluginVersion": "7.5.3",
+ "targets": [
+ {
+ "datasource": {
+ "uid": "${datasource}"
+ },
+ "editorMode": "code",
+ "exemplar": true,
+ "expr": "sum(mysql_global_status_innodb_buffer_pool_bytes_data{app_kubernetes_io_instance=\"$polardbx\", app_kubernetes_io_component=\"gms\", namespace=\"$namespace\"})",
+ "interval": "",
+ "legendFormat": "buffer pool size",
+ "queryType": "randomWalk",
+ "range": true,
+ "refId": "A"
+ }
+ ],
+ "title": "Buffer Pool Size (Total)",
+ "type": "timeseries"
+ },
+ {
+ "datasource": {
+ "type": "prometheus",
+ "uid": "${datasource}"
+ },
+ "fieldConfig": {
+ "defaults": {
+ "color": {
+ "mode": "palette-classic"
+ },
+ "custom": {
+ "axisCenteredZero": false,
+ "axisColorMode": "text",
+ "axisLabel": "",
+ "axisPlacement": "auto",
+ "barAlignment": 0,
+ "drawStyle": "line",
+ "fillOpacity": 10,
+ "gradientMode": "opacity",
+ "hideFrom": {
+ "graph": false,
+ "legend": false,
+ "tooltip": false,
+ "viz": false
+ },
+ "lineInterpolation": "linear",
+ "lineStyle": {
+ "fill": "solid"
+ },
+ "lineWidth": 1,
+ "pointSize": 5,
+ "scaleDistribution": {
+ "type": "linear"
+ },
+ "showPoints": "never",
+ "spanNulls": true,
+ "stacking": {
+ "group": "A",
+ "mode": "none"
+ },
+ "thresholdsStyle": {
+ "mode": "off"
+ }
+ },
+ "mappings": [],
+ "thresholds": {
+ "mode": "absolute",
+ "steps": [
+ {
+ "color": "green",
+ "value": null
+ }
+ ]
+ },
+ "unit": "bytes"
+ },
+ "overrides": []
+ },
+ "gridPos": {
+ "h": 8,
+ "w": 6,
+ "x": 18,
+ "y": 20
+ },
+ "id": 89,
+ "options": {
+ "graph": {},
+ "legend": {
+ "calcs": [],
+ "displayMode": "list",
+ "placement": "bottom",
+ "showLegend": true
+ },
+ "tooltip": {
+ "mode": "multi",
+ "sort": "none"
+ }
+ },
+ "pluginVersion": "7.5.3",
+ "targets": [
+ {
+ "datasource": {
+ "uid": "${datasource}"
+ },
+ "editorMode": "code",
+ "exemplar": true,
+ "expr": "sum(mysql_global_status_innodb_buffer_pool_bytes_data{app_kubernetes_io_instance=\"$polardbx\", app_kubernetes_io_component=\"gms\", namespace=\"$namespace\"}) by (xstore_name)",
+ "interval": "",
+ "legendFormat": "{{ xstore_name }}",
+ "queryType": "randomWalk",
+ "range": true,
+ "refId": "A"
+ }
+ ],
+ "title": "Buffer Pool Size (GMS)",
+ "type": "timeseries"
+ },
+ {
+ "datasource": {
+ "type": "prometheus",
+ "uid": "${datasource}"
+ },
+ "fieldConfig": {
+ "defaults": {
+ "color": {
+ "mode": "palette-classic"
+ },
+ "custom": {
+ "axisCenteredZero": false,
+ "axisColorMode": "text",
+ "axisLabel": "",
+ "axisPlacement": "auto",
+ "barAlignment": 0,
+ "drawStyle": "line",
+ "fillOpacity": 10,
+ "gradientMode": "opacity",
+ "hideFrom": {
+ "graph": false,
+ "legend": false,
+ "tooltip": false,
+ "viz": false
+ },
+ "lineInterpolation": "linear",
+ "lineStyle": {
+ "fill": "solid"
+ },
+ "lineWidth": 1,
+ "pointSize": 5,
+ "scaleDistribution": {
+ "log": 10,
+ "type": "log"
+ },
+ "showPoints": "never",
+ "spanNulls": true,
+ "stacking": {
+ "group": "A",
+ "mode": "none"
+ },
+ "thresholdsStyle": {
+ "mode": "off"
+ }
+ },
+ "mappings": [],
+ "thresholds": {
+ "mode": "absolute",
+ "steps": [
+ {
+ "color": "green",
+ "value": null
+ }
+ ]
+ },
+ "unit": "short"
+ },
+ "overrides": []
+ },
+ "gridPos": {
+ "h": 8,
+ "w": 6,
+ "x": 0,
+ "y": 28
+ },
+ "id": 90,
+ "options": {
+ "graph": {},
+ "legend": {
+ "calcs": [],
+ "displayMode": "list",
+ "placement": "bottom",
+ "showLegend": true
+ },
+ "tooltip": {
+ "mode": "multi",
+ "sort": "none"
+ }
+ },
+ "pluginVersion": "7.5.3",
+ "targets": [
+ {
+ "datasource": {
+ "uid": "${datasource}"
+ },
+ "editorMode": "code",
+ "exemplar": true,
+ "expr": "sum(mysql_global_status_buffer_pool_dirty_pages{app_kubernetes_io_instance=\"$polardbx\", app_kubernetes_io_component=\"gms\", namespace=\"$namespace\"})",
+ "interval": "",
+ "legendFormat": "dirty",
+ "queryType": "randomWalk",
+ "range": true,
+ "refId": "A"
+ },
+ {
+ "datasource": {
+ "uid": "${datasource}"
+ },
+ "editorMode": "code",
+ "exemplar": true,
+ "expr": "sum(rate(mysql_global_status_buffer_pool_page_changes_total{app_kubernetes_io_instance=\"$polardbx\", app_kubernetes_io_component=\"gms\", namespace=\"$namespace\"}[10m]))",
+ "hide": false,
+ "interval": "",
+ "legendFormat": "flush",
+ "queryType": "randomWalk",
+ "range": true,
+ "refId": "B"
+ }
+ ],
+ "title": "Buffer Pool Dirty/Flush (Total)",
+ "type": "timeseries"
+ },
+ {
+ "datasource": {
+ "type": "prometheus",
+ "uid": "${datasource}"
+ },
+ "fieldConfig": {
+ "defaults": {
+ "color": {
+ "mode": "palette-classic"
+ },
+ "custom": {
+ "axisCenteredZero": false,
+ "axisColorMode": "text",
+ "axisLabel": "",
+ "axisPlacement": "auto",
+ "barAlignment": 0,
+ "drawStyle": "line",
+ "fillOpacity": 10,
+ "gradientMode": "opacity",
+ "hideFrom": {
+ "graph": false,
+ "legend": false,
+ "tooltip": false,
+ "viz": false
+ },
+ "lineInterpolation": "linear",
+ "lineStyle": {
+ "fill": "solid"
+ },
+ "lineWidth": 1,
+ "pointSize": 5,
+ "scaleDistribution": {
+ "type": "linear"
+ },
+ "showPoints": "never",
+ "spanNulls": true,
+ "stacking": {
+ "group": "A",
+ "mode": "none"
+ },
+ "thresholdsStyle": {
+ "mode": "off"
+ }
+ },
+ "mappings": [],
+ "thresholds": {
+ "mode": "absolute",
+ "steps": [
+ {
+ "color": "green",
+ "value": null
+ }
+ ]
+ },
+ "unit": "short"
+ },
+ "overrides": []
+ },
+ "gridPos": {
+ "h": 8,
+ "w": 6,
+ "x": 6,
+ "y": 28
+ },
+ "id": 91,
+ "options": {
+ "graph": {},
+ "legend": {
+ "calcs": [],
+ "displayMode": "list",
+ "placement": "bottom",
+ "showLegend": true
+ },
+ "tooltip": {
+ "mode": "single",
+ "sort": "none"
+ }
+ },
+ "pluginVersion": "7.5.3",
+ "targets": [
+ {
+ "datasource": {
+ "uid": "${datasource}"
+ },
+ "editorMode": "code",
+ "exemplar": true,
+ "expr": "sum(mysql_global_status_innodb_row_lock_current_waits{app_kubernetes_io_instance=\"$polardbx\", app_kubernetes_io_component=\"gms\", namespace=\"$namespace\"})",
+ "interval": "",
+ "legendFormat": "current wait",
+ "queryType": "randomWalk",
+ "range": true,
+ "refId": "A"
+ }
+ ],
+ "title": "Row Lock Current Wait (Total)",
+ "type": "timeseries"
+ },
+ {
+ "datasource": {
+ "type": "prometheus",
+ "uid": "${datasource}"
+ },
+ "fieldConfig": {
+ "defaults": {
+ "color": {
+ "mode": "palette-classic"
+ },
+ "custom": {
+ "axisCenteredZero": false,
+ "axisColorMode": "text",
+ "axisLabel": "",
+ "axisPlacement": "auto",
+ "barAlignment": 0,
+ "drawStyle": "line",
+ "fillOpacity": 10,
+ "gradientMode": "opacity",
+ "hideFrom": {
+ "graph": false,
+ "legend": false,
+ "tooltip": false,
+ "viz": false
+ },
+ "lineInterpolation": "linear",
+ "lineStyle": {
+ "fill": "solid"
+ },
+ "lineWidth": 1,
+ "pointSize": 5,
+ "scaleDistribution": {
+ "type": "linear"
+ },
+ "showPoints": "never",
+ "spanNulls": true,
+ "stacking": {
+ "group": "A",
+ "mode": "none"
+ },
+ "thresholdsStyle": {
+ "mode": "off"
+ }
+ },
+ "mappings": [],
+ "thresholds": {
+ "mode": "absolute",
+ "steps": [
+ {
+ "color": "green",
+ "value": null
+ }
+ ]
+ },
+ "unit": "short"
+ },
+ "overrides": []
+ },
+ "gridPos": {
+ "h": 8,
+ "w": 6,
+ "x": 12,
+ "y": 28
+ },
+ "id": 92,
+ "options": {
+ "graph": {},
+ "legend": {
+ "calcs": [],
+ "displayMode": "list",
+ "placement": "bottom",
+ "showLegend": true
+ },
+ "tooltip": {
+ "mode": "multi",
+ "sort": "none"
+ }
+ },
+ "pluginVersion": "7.5.3",
+ "targets": [
+ {
+ "datasource": {
+ "uid": "${datasource}"
+ },
+ "editorMode": "code",
+ "exemplar": true,
+ "expr": "sum(rate(mysql_global_status_innodb_buffer_pool_read_requests{app_kubernetes_io_instance=\"$polardbx\", app_kubernetes_io_component=\"gms\", namespace=\"$namespace\"}[1m]))",
+ "interval": "",
+ "legendFormat": "reads",
+ "queryType": "randomWalk",
+ "range": true,
+ "refId": "A"
+ },
+ {
+ "datasource": {
+ "uid": "${datasource}"
+ },
+ "editorMode": "code",
+ "exemplar": true,
+ "expr": "sum(rate(mysql_global_status_innodb_buffer_pool_write_requests{app_kubernetes_io_instance=\"$polardbx\", app_kubernetes_io_component=\"gms\", namespace=\"$namespace\"}[1m]))",
+ "hide": false,
+ "interval": "",
+ "legendFormat": "writes",
+ "queryType": "randomWalk",
+ "range": true,
+ "refId": "B"
+ }
+ ],
+ "title": "Logical Reads/Writes (Total)",
+ "type": "timeseries"
+ },
+ {
+ "datasource": {
+ "type": "prometheus",
+ "uid": "${datasource}"
+ },
+ "fieldConfig": {
+ "defaults": {
+ "color": {
+ "mode": "palette-classic"
+ },
+ "custom": {
+ "axisCenteredZero": false,
+ "axisColorMode": "text",
+ "axisLabel": "",
+ "axisPlacement": "auto",
+ "barAlignment": 0,
+ "drawStyle": "line",
+ "fillOpacity": 10,
+ "gradientMode": "opacity",
+ "hideFrom": {
+ "graph": false,
+ "legend": false,
+ "tooltip": false,
+ "viz": false
+ },
+ "lineInterpolation": "linear",
+ "lineStyle": {
+ "fill": "solid"
+ },
+ "lineWidth": 1,
+ "pointSize": 5,
+ "scaleDistribution": {
+ "type": "linear"
+ },
+ "showPoints": "never",
+ "spanNulls": true,
+ "stacking": {
+ "group": "A",
+ "mode": "none"
+ },
+ "thresholdsStyle": {
+ "mode": "off"
+ }
+ },
+ "mappings": [],
+ "thresholds": {
+ "mode": "absolute",
+ "steps": [
+ {
+ "color": "green",
+ "value": null
+ }
+ ]
+ },
+ "unit": "short"
+ },
+ "overrides": []
+ },
+ "gridPos": {
+ "h": 8,
+ "w": 6,
+ "x": 18,
+ "y": 28
+ },
+ "id": 93,
+ "options": {
+ "graph": {},
+ "legend": {
+ "calcs": [],
+ "displayMode": "list",
+ "placement": "bottom",
+ "showLegend": true
+ },
+ "tooltip": {
+ "mode": "multi",
+ "sort": "none"
+ }
+ },
+ "pluginVersion": "7.5.3",
+ "targets": [
+ {
+ "datasource": {
+ "uid": "${datasource}"
+ },
+ "editorMode": "code",
+ "exemplar": true,
+ "expr": "sum(rate(mysql_global_status_innodb_log_writes{app_kubernetes_io_instance=\"$polardbx\", app_kubernetes_io_component=\"gms\", namespace=\"$namespace\"}[1m]))",
+ "interval": "",
+ "legendFormat": "log writes",
+ "queryType": "randomWalk",
+ "range": true,
+ "refId": "A"
+ },
+ {
+ "datasource": {
+ "uid": "${datasource}"
+ },
+ "editorMode": "code",
+ "exemplar": true,
+ "expr": "sum(rate(mysql_global_status_innodb_data_reads{app_kubernetes_io_instance=\"$polardbx\", app_kubernetes_io_component=\"gms\", namespace=\"$namespace\"}[1m]))",
+ "hide": false,
+ "interval": "",
+ "legendFormat": "data reads",
+ "queryType": "randomWalk",
+ "range": true,
+ "refId": "B"
+ },
+ {
+ "datasource": {
+ "uid": "${datasource}"
+ },
+ "exemplar": true,
+ "expr": "sum(rate(mysql_global_status_innodb_data_fsyncs{polardbx_name=\"$polardbx\", polardbx_role=\"dn\", xstore_role=\"leader\", namespace=\"$namespace\"}[1m]))",
+ "hide": false,
+ "interval": "",
+ "legendFormat": "data fsyncs",
+ "queryType": "randomWalk",
+ "refId": "C"
+ },
+ {
+ "datasource": {
+ "uid": "${datasource}"
+ },
+ "exemplar": true,
+ "expr": "sum(rate(mysql_global_status_innodb_os_log_fsyncs{polardbx_name=\"$polardbx\", polardbx_role=\"dn\", xstore_role=\"leader\", namespace=\"$namespace\"}[1m]))",
+ "hide": false,
+ "interval": "",
+ "legendFormat": "log fsyncs",
+ "queryType": "randomWalk",
+ "refId": "D"
+ }
+ ],
+ "title": "IOPS (Total)",
+ "type": "timeseries"
+ },
+ {
+ "datasource": {
+ "type": "prometheus",
+ "uid": "${datasource}"
+ },
+ "fieldConfig": {
+ "defaults": {
+ "color": {
+ "mode": "palette-classic"
+ },
+ "custom": {
+ "axisCenteredZero": false,
+ "axisColorMode": "text",
+ "axisLabel": "",
+ "axisPlacement": "auto",
+ "barAlignment": 0,
+ "drawStyle": "line",
+ "fillOpacity": 10,
+ "gradientMode": "opacity",
+ "hideFrom": {
+ "graph": false,
+ "legend": false,
+ "tooltip": false,
+ "viz": false
+ },
+ "lineInterpolation": "linear",
+ "lineStyle": {
+ "fill": "solid"
+ },
+ "lineWidth": 1,
+ "pointSize": 5,
+ "scaleDistribution": {
+ "type": "linear"
+ },
+ "showPoints": "never",
+ "spanNulls": true,
+ "stacking": {
+ "group": "A",
+ "mode": "none"
+ },
+ "thresholdsStyle": {
+ "mode": "off"
+ }
+ },
+ "mappings": [],
+ "thresholds": {
+ "mode": "absolute",
+ "steps": [
+ {
+ "color": "green"
+ }
+ ]
+ },
+ "unit": "bytes"
+ },
+ "overrides": []
+ },
+ "gridPos": {
+ "h": 8,
+ "w": 6,
+ "x": 0,
+ "y": 36
+ },
+ "id": 94,
+ "options": {
+ "graph": {},
+ "legend": {
+ "calcs": [],
+ "displayMode": "list",
+ "placement": "bottom",
+ "showLegend": true
+ },
+ "tooltip": {
+ "mode": "multi",
+ "sort": "none"
+ }
+ },
+ "pluginVersion": "7.5.3",
+ "targets": [
+ {
+ "datasource": {
+ "uid": "${datasource}"
+ },
+ "editorMode": "code",
+ "exemplar": true,
+ "expr": "sum(rate(mysql_global_status_innodb_data_written{app_kubernetes_io_instance=\"$polardbx\", app_kubernetes_io_component=\"gms\", namespace=\"$namespace\"}[1m]))",
+ "interval": "",
+ "legendFormat": "data written/s",
+ "queryType": "randomWalk",
+ "range": true,
+ "refId": "A"
+ },
+ {
+ "datasource": {
+ "uid": "${datasource}"
+ },
+ "editorMode": "code",
+ "exemplar": true,
+ "expr": "sum(rate(mysql_global_status_innodb_data_read{app_kubernetes_io_instance=\"$polardbx\", app_kubernetes_io_component=\"gms\", namespace=\"$namespace\"}[1m]))",
+ "hide": false,
+ "interval": "",
+ "legendFormat": "data read/s",
+ "queryType": "randomWalk",
+ "range": true,
+ "refId": "B"
+ },
+ {
+ "datasource": {
+ "uid": "${datasource}"
+ },
+ "exemplar": true,
+ "expr": "sum(rate(mysql_global_status_innodb_os_log_written{polardbx_name=\"$polardbx\", polardbx_role=\"dn\", xstore_role=\"leader\", namespace=\"$namespace\"}[1m]))",
+ "hide": false,
+ "interval": "",
+ "legendFormat": "log writtern/s",
+ "queryType": "randomWalk",
+ "refId": "C"
+ }
+ ],
+ "title": "IO Throughput (Total)",
+ "type": "timeseries"
+ },
+ {
+ "datasource": {
+ "type": "prometheus",
+ "uid": "${datasource}"
+ },
+ "fieldConfig": {
+ "defaults": {
+ "color": {
+ "mode": "palette-classic"
+ },
+ "custom": {
+ "axisCenteredZero": false,
+ "axisColorMode": "text",
+ "axisLabel": "",
+ "axisPlacement": "auto",
+ "barAlignment": 0,
+ "drawStyle": "line",
+ "fillOpacity": 10,
+ "gradientMode": "opacity",
+ "hideFrom": {
+ "graph": false,
+ "legend": false,
+ "tooltip": false,
+ "viz": false
+ },
+ "lineInterpolation": "linear",
+ "lineStyle": {
+ "fill": "solid"
+ },
+ "lineWidth": 1,
+ "pointSize": 5,
+ "scaleDistribution": {
+ "type": "linear"
+ },
+ "showPoints": "never",
+ "spanNulls": true,
+ "stacking": {
+ "group": "A",
+ "mode": "none"
+ },
+ "thresholdsStyle": {
+ "mode": "off"
+ }
+ },
+ "mappings": [],
+ "thresholds": {
+ "mode": "absolute",
+ "steps": [
+ {
+ "color": "green"
+ }
+ ]
+ },
+ "unit": "bytes"
+ },
+ "overrides": []
+ },
+ "gridPos": {
+ "h": 8,
+ "w": 6,
+ "x": 6,
+ "y": 36
+ },
+ "id": 95,
+ "options": {
+ "graph": {},
+ "legend": {
+ "calcs": [],
+ "displayMode": "list",
+ "placement": "bottom",
+ "showLegend": true
+ },
+ "tooltip": {
+ "mode": "multi",
+ "sort": "none"
+ }
+ },
+ "pluginVersion": "7.5.3",
+ "targets": [
+ {
+ "datasource": {
+ "uid": "${datasource}"
+ },
+ "editorMode": "code",
+ "exemplar": true,
+ "expr": "sum(rate(mysql_global_status_bytes_received{app_kubernetes_io_instance=\"$polardbx\", app_kubernetes_io_component=\"gms\", namespace=\"$namespace\"}[1m]))",
+ "interval": "",
+ "legendFormat": "received/s",
+ "queryType": "randomWalk",
+ "range": true,
+ "refId": "A"
+ },
+ {
+ "datasource": {
+ "uid": "${datasource}"
+ },
+ "editorMode": "code",
+ "exemplar": true,
+ "expr": "sum(rate(mysql_global_status_bytes_sent{app_kubernetes_io_instance=\"$polardbx\", app_kubernetes_io_component=\"gms\", namespace=\"$namespace\"}[1m]))",
+ "hide": false,
+ "interval": "",
+ "legendFormat": "sent/s",
+ "queryType": "randomWalk",
+ "range": true,
+ "refId": "B"
+ }
+ ],
+ "title": "Network (Total)",
+ "type": "timeseries"
+ },
+ {
+ "collapsed": false,
+ "datasource": {
+ "type": "prometheus",
+ "uid": "prometheus"
+ },
+ "gridPos": {
+ "h": 1,
+ "w": 24,
+ "x": 0,
+ "y": 44
+ },
+ "id": 4,
+ "panels": [],
+ "targets": [
+ {
+ "datasource": {
+ "type": "prometheus",
+ "uid": "prometheus"
+ },
+ "refId": "A"
+ }
+ ],
+ "title": "Compute Node",
+ "type": "row"
+ },
+ {
+ "datasource": {
+ "type": "prometheus",
+ "uid": "${datasource}"
+ },
+ "fieldConfig": {
+ "defaults": {
+ "color": {
+ "mode": "palette-classic"
+ },
+ "custom": {
+ "axisCenteredZero": false,
+ "axisColorMode": "text",
+ "axisLabel": "",
+ "axisPlacement": "auto",
+ "barAlignment": 0,
+ "drawStyle": "line",
+ "fillOpacity": 10,
+ "gradientMode": "opacity",
+ "hideFrom": {
+ "graph": false,
+ "legend": false,
+ "tooltip": false,
+ "viz": false
+ },
+ "lineInterpolation": "linear",
+ "lineStyle": {
+ "fill": "solid"
+ },
+ "lineWidth": 1,
+ "pointSize": 5,
+ "scaleDistribution": {
+ "type": "linear"
+ },
+ "showPoints": "never",
+ "spanNulls": true,
+ "stacking": {
+ "group": "A",
+ "mode": "none"
+ },
+ "thresholdsStyle": {
+ "mode": "off"
+ }
+ },
+ "mappings": [],
+ "thresholds": {
+ "mode": "absolute",
+ "steps": [
+ {
+ "color": "green"
+ }
+ ]
+ },
+ "unit": "short"
+ },
+ "overrides": []
+ },
+ "gridPos": {
+ "h": 8,
+ "w": 6,
+ "x": 0,
+ "y": 45
+ },
+ "id": 28,
+ "options": {
+ "graph": {},
+ "legend": {
+ "calcs": [],
+ "displayMode": "list",
+ "placement": "bottom",
+ "showLegend": true
+ },
+ "tooltip": {
+ "mode": "multi",
+ "sort": "none"
+ }
+ },
+ "pluginVersion": "7.5.3",
+ "targets": [
+ {
+ "datasource": {
+ "uid": "${datasource}"
+ },
+ "editorMode": "code",
+ "exemplar": true,
+ "expr": "sum(rate(polardbx_stats_request_count_total{app_kubernetes_io_instance=\"$polardbx\", namespace=\"$namespace\"}[1m]))",
+ "interval": "",
+ "legendFormat": "qps (logical)",
+ "queryType": "randomWalk",
+ "range": true,
+ "refId": "A"
+ },
+ {
+ "datasource": {
+ "uid": "${datasource}"
+ },
+ "editorMode": "code",
+ "exemplar": true,
+ "expr": "sum(rate(polardbx_stats_physical_request_count_total{app_kubernetes_io_instance=\"$polardbx\", namespace=\"$namespace\"}[1m]))",
+ "hide": false,
+ "interval": "",
+ "legendFormat": "qps (physical)",
+ "range": true,
+ "refId": "B"
+ }
+ ],
+ "title": "QPS",
+ "type": "timeseries"
+ },
+ {
+ "datasource": {
+ "type": "prometheus",
+ "uid": "${datasource}"
+ },
+ "fieldConfig": {
+ "defaults": {
+ "color": {
+ "mode": "palette-classic"
+ },
+ "custom": {
+ "axisCenteredZero": false,
+ "axisColorMode": "text",
+ "axisLabel": "",
+ "axisPlacement": "auto",
+ "barAlignment": 0,
+ "drawStyle": "line",
+ "fillOpacity": 10,
+ "gradientMode": "opacity",
+ "hideFrom": {
+ "graph": false,
+ "legend": false,
+ "tooltip": false,
+ "viz": false
+ },
+ "lineInterpolation": "linear",
+ "lineStyle": {
+ "fill": "solid"
+ },
+ "lineWidth": 1,
+ "pointSize": 5,
+ "scaleDistribution": {
+ "type": "linear"
+ },
+ "showPoints": "never",
+ "spanNulls": true,
+ "stacking": {
+ "group": "A",
+ "mode": "none"
+ },
+ "thresholdsStyle": {
+ "mode": "off"
+ }
+ },
+ "decimals": 2,
+ "mappings": [],
+ "thresholds": {
+ "mode": "absolute",
+ "steps": [
+ {
+ "color": "green"
+ }
+ ]
+ },
+ "unit": "ms"
+ },
+ "overrides": []
+ },
+ "gridPos": {
+ "h": 8,
+ "w": 6,
+ "x": 6,
+ "y": 45
+ },
+ "id": 29,
+ "options": {
+ "graph": {},
+ "legend": {
+ "calcs": [],
+ "displayMode": "list",
+ "placement": "bottom",
+ "showLegend": true
+ },
+ "tooltip": {
+ "mode": "multi",
+ "sort": "none"
+ }
+ },
+ "pluginVersion": "7.5.3",
+ "targets": [
+ {
+ "datasource": {
+ "uid": "${datasource}"
+ },
+ "editorMode": "code",
+ "exemplar": true,
+ "expr": "sum(delta(polardbx_stats_request_time_cost_total{app_kubernetes_io_instance=\"$polardbx\", namespace=\"$namespace\"}[1m])) / (sum(delta (polardbx_stats_request_count_total{app_kubernetes_io_instance=\"$polardbx\", namespace=\"$namespace\"}[1m])) + 1) / 1000",
+ "interval": "",
+ "legendFormat": "rt (logical)",
+ "queryType": "randomWalk",
+ "range": true,
+ "refId": "A"
+ },
+ {
+ "datasource": {
+ "uid": "${datasource}"
+ },
+ "editorMode": "code",
+ "exemplar": true,
+ "expr": "sum(delta(polardbx_stats_physical_request_time_cost_total{app_kubernetes_io_instance=\"$polardbx\", namespace=\"$namespace\"}[30s])) / (sum(delta (polardbx_stats_physical_request_count_total{app_kubernetes_io_instance=\"$polardbx\", namespace=\"$namespace\"}[30s])) + 1) / 1000",
+ "hide": false,
+ "interval": "",
+ "legendFormat": "rt (physical)",
+ "range": true,
+ "refId": "B"
+ }
+ ],
+ "title": "Response Time",
+ "type": "timeseries"
+ },
+ {
+ "datasource": {
+ "type": "prometheus",
+ "uid": "${datasource}"
+ },
+ "fieldConfig": {
+ "defaults": {
+ "color": {
+ "mode": "palette-classic"
+ },
+ "custom": {
+ "axisCenteredZero": false,
+ "axisColorMode": "text",
+ "axisLabel": "",
+ "axisPlacement": "auto",
+ "barAlignment": 0,
+ "drawStyle": "line",
+ "fillOpacity": 10,
+ "gradientMode": "opacity",
+ "hideFrom": {
+ "graph": false,
+ "legend": false,
+ "tooltip": false,
+ "viz": false
+ },
+ "lineInterpolation": "linear",
+ "lineStyle": {
+ "fill": "solid"
+ },
+ "lineWidth": 1,
+ "pointSize": 5,
+ "scaleDistribution": {
+ "type": "linear"
+ },
+ "showPoints": "never",
+ "spanNulls": true,
+ "stacking": {
+ "group": "A",
+ "mode": "none"
+ },
+ "thresholdsStyle": {
+ "mode": "off"
+ }
+ },
+ "mappings": [],
+ "thresholds": {
+ "mode": "absolute",
+ "steps": [
+ {
+ "color": "green"
+ }
+ ]
+ },
+ "unit": "short"
+ },
+ "overrides": []
+ },
+ "gridPos": {
+ "h": 8,
+ "w": 6,
+ "x": 12,
+ "y": 45
+ },
+ "id": 30,
+ "options": {
+ "graph": {},
+ "legend": {
+ "calcs": [],
+ "displayMode": "list",
+ "placement": "bottom",
+ "showLegend": true
+ },
+ "tooltip": {
+ "mode": "multi",
+ "sort": "none"
+ }
+ },
+ "pluginVersion": "7.5.3",
+ "targets": [
+ {
+ "datasource": {
+ "uid": "${datasource}"
+ },
+ "editorMode": "code",
+ "exemplar": true,
+ "expr": "sum(polardbx_stats_active_connections{app_kubernetes_io_instance=\"$polardbx\", namespace=\"$namespace\"})",
+ "interval": "",
+ "legendFormat": "connections",
+ "queryType": "randomWalk",
+ "range": true,
+ "refId": "A"
+ },
+ {
+ "datasource": {
+ "uid": "${datasource}"
+ },
+ "editorMode": "code",
+ "exemplar": true,
+ "expr": "sum(polardbx_stats_running_count{app_kubernetes_io_instance=\"$polardbx\", namespace=\"$namespace\"})",
+ "hide": false,
+ "interval": "",
+ "legendFormat": "threads (running)",
+ "range": true,
+ "refId": "B"
+ }
+ ],
+ "title": "Connections / Threads",
+ "type": "timeseries"
+ },
+ {
+ "datasource": {
+ "type": "prometheus",
+ "uid": "${datasource}"
+ },
+ "fieldConfig": {
+ "defaults": {
+ "color": {
+ "mode": "palette-classic"
+ },
+ "custom": {
+ "axisCenteredZero": false,
+ "axisColorMode": "text",
+ "axisLabel": "",
+ "axisPlacement": "auto",
+ "barAlignment": 0,
+ "drawStyle": "line",
+ "fillOpacity": 10,
+ "gradientMode": "none",
+ "hideFrom": {
+ "graph": false,
+ "legend": false,
+ "tooltip": false,
+ "viz": false
+ },
+ "lineInterpolation": "linear",
+ "lineWidth": 1,
+ "pointSize": 5,
+ "scaleDistribution": {
+ "type": "linear"
+ },
+ "showPoints": "never",
+ "spanNulls": true,
+ "stacking": {
+ "group": "A",
+ "mode": "none"
+ },
+ "thresholdsStyle": {
+ "mode": "off"
+ }
+ },
+ "mappings": [],
+ "thresholds": {
+ "mode": "absolute",
+ "steps": [
+ {
+ "color": "green"
+ },
+ {
+ "color": "red",
+ "value": 80
+ }
+ ]
+ },
+ "unit": "ns"
+ },
+ "overrides": [
+ {
+ "matcher": {
+ "id": "byFrameRefID",
+ "options": "A"
+ },
+ "properties": [
+ {
+ "id": "custom.axisPlacement",
+ "value": "right"
+ },
+ {
+ "id": "custom.axisLabel",
+ "value": "count"
+ },
+ {
+ "id": "unit",
+ "value": "short"
+ },
+ {
+ "id": "custom.fillOpacity",
+ "value": 0
+ },
+ {
+ "id": "custom.gradientMode",
+ "value": "none"
+ }
+ ]
+ },
+ {
+ "matcher": {
+ "id": "byFrameRefID",
+ "options": "B"
+ },
+ "properties": [
+ {
+ "id": "custom.axisPlacement",
+ "value": "left"
+ },
+ {
+ "id": "unit",
+ "value": "ns"
+ },
+ {
+ "id": "custom.axisLabel",
+ "value": "time"
+ },
+ {
+ "id": "custom.fillOpacity",
+ "value": 0
+ },
+ {
+ "id": "custom.gradientMode",
+ "value": "none"
+ }
+ ]
+ }
+ ]
+ },
+ "gridPos": {
+ "h": 8,
+ "w": 6,
+ "x": 18,
+ "y": 45
+ },
+ "id": 31,
+ "options": {
+ "graph": {},
+ "legend": {
+ "calcs": [],
+ "displayMode": "list",
+ "placement": "bottom",
+ "showLegend": true
+ },
+ "tooltip": {
+ "mode": "multi",
+ "sort": "none"
+ }
+ },
+ "pluginVersion": "7.5.3",
+ "targets": [
+ {
+ "datasource": {
+ "uid": "${datasource}"
+ },
+ "editorMode": "code",
+ "exemplar": true,
+ "expr": "sum(idelta(polardbx_jvmstats_gc_collector_invocation_count{app_kubernetes_io_instance=\"$polardbx\", namespace=\"$namespace\"}[1m])) by (type)",
+ "instant": false,
+ "interval": "",
+ "legendFormat": "count - {{ type }}",
+ "queryType": "randomWalk",
+ "refId": "A"
+ },
+ {
+ "datasource": {
+ "uid": "${datasource}"
+ },
+ "editorMode": "code",
+ "exemplar": true,
+ "expr": "sum(idelta(polardbx_jvmstats_gc_collector_time_total{app_kubernetes_io_instance=\"$polardbx\", namespace=\"$namespace\"}[1m]) / (idelta(polardbx_jvmstats_gc_collector_invocation_count{app_kubernetes_io_instance=\"$polardbx\", namespace=\"$namespace\"}[1m]) + 1)) by (type)",
+ "hide": false,
+ "interval": "",
+ "legendFormat": "time - {{ type }}",
+ "range": true,
+ "refId": "B"
+ }
+ ],
+ "title": "GC",
+ "type": "timeseries"
+ },
+ {
+ "datasource": {
+ "type": "prometheus",
+ "uid": "${datasource}"
+ },
+ "fieldConfig": {
+ "defaults": {
+ "color": {
+ "mode": "palette-classic"
+ },
+ "custom": {
+ "axisCenteredZero": false,
+ "axisColorMode": "text",
+ "axisLabel": "",
+ "axisPlacement": "auto",
+ "barAlignment": 0,
+ "drawStyle": "line",
+ "fillOpacity": 10,
+ "gradientMode": "hue",
+ "hideFrom": {
+ "graph": false,
+ "legend": false,
+ "tooltip": false,
+ "viz": false
+ },
+ "lineInterpolation": "linear",
+ "lineStyle": {
+ "fill": "solid"
+ },
+ "lineWidth": 1,
+ "pointSize": 5,
+ "scaleDistribution": {
+ "type": "linear"
+ },
+ "showPoints": "never",
+ "spanNulls": true,
+ "stacking": {
+ "group": "A",
+ "mode": "none"
+ },
+ "thresholdsStyle": {
+ "mode": "off"
+ }
+ },
+ "mappings": [],
+ "thresholds": {
+ "mode": "absolute",
+ "steps": [
+ {
+ "color": "green"
+ }
+ ]
+ },
+ "unit": "short"
+ },
+ "overrides": []
+ },
+ "gridPos": {
+ "h": 8,
+ "w": 6,
+ "x": 0,
+ "y": 53
+ },
+ "id": 32,
+ "options": {
+ "graph": {},
+ "legend": {
+ "calcs": [],
+ "displayMode": "list",
+ "placement": "bottom",
+ "showLegend": true
+ },
+ "tooltip": {
+ "mode": "single",
+ "sort": "none"
+ }
+ },
+ "pluginVersion": "7.5.3",
+ "targets": [
+ {
+ "datasource": {
+ "uid": "${datasource}"
+ },
+ "editorMode": "code",
+ "exemplar": true,
+ "expr": "sum(rate(polardbx_stc_connection_error_count_total{app_kubernetes_io_instance=\"$polardbx\", namespace=\"$namespace\"}[1m]))",
+ "interval": "",
+ "legendFormat": "connection error",
+ "queryType": "randomWalk",
+ "range": true,
+ "refId": "A"
+ }
+ ],
+ "title": "Connection Error",
+ "type": "timeseries"
+ },
+ {
+ "datasource": {
+ "type": "prometheus",
+ "uid": "${datasource}"
+ },
+ "fieldConfig": {
+ "defaults": {
+ "color": {
+ "mode": "palette-classic"
+ },
+ "custom": {
+ "axisCenteredZero": false,
+ "axisColorMode": "text",
+ "axisLabel": "",
+ "axisPlacement": "auto",
+ "barAlignment": 0,
+ "drawStyle": "line",
+ "fillOpacity": 10,
+ "gradientMode": "hue",
+ "hideFrom": {
+ "graph": false,
+ "legend": false,
+ "tooltip": false,
+ "viz": false
+ },
+ "lineInterpolation": "linear",
+ "lineStyle": {
+ "fill": "solid"
+ },
+ "lineWidth": 1,
+ "pointSize": 5,
+ "scaleDistribution": {
+ "type": "linear"
+ },
+ "showPoints": "never",
+ "spanNulls": true,
+ "stacking": {
+ "group": "A",
+ "mode": "none"
+ },
+ "thresholdsStyle": {
+ "mode": "off"
+ }
+ },
+ "mappings": [],
+ "thresholds": {
+ "mode": "absolute",
+ "steps": [
+ {
+ "color": "green"
+ }
+ ]
+ },
+ "unit": "short"
+ },
+ "overrides": []
+ },
+ "gridPos": {
+ "h": 8,
+ "w": 6,
+ "x": 6,
+ "y": 53
+ },
+ "id": 39,
+ "options": {
+ "graph": {},
+ "legend": {
+ "calcs": [],
+ "displayMode": "list",
+ "placement": "bottom",
+ "showLegend": true
+ },
+ "tooltip": {
+ "mode": "multi",
+ "sort": "none"
+ }
+ },
+ "pluginVersion": "7.5.3",
+ "targets": [
+ {
+ "datasource": {
+ "uid": "${datasource}"
+ },
+ "editorMode": "code",
+ "exemplar": true,
+ "expr": "sum(rate(polardbx_stats_best_effort_transaction_count_total{app_kubernetes_io_instance=\"$polardbx\", namespace=\"$namespace\"}[1m]))",
+ "interval": "",
+ "legendFormat": "best effort",
+ "queryType": "randomWalk",
+ "range": true,
+ "refId": "A"
+ },
+ {
+ "datasource": {
+ "uid": "${datasource}"
+ },
+ "editorMode": "code",
+ "exemplar": true,
+ "expr": "sum(rate(polardbx_stats_xa_transaction_count_total{app_kubernetes_io_instance=\"$polardbx\", namespace=\"$namespace\"}[1m]))",
+ "hide": false,
+ "interval": "",
+ "legendFormat": "xa",
+ "queryType": "randomWalk",
+ "range": true,
+ "refId": "B"
+ },
+ {
+ "datasource": {
+ "uid": "${datasource}"
+ },
+ "exemplar": true,
+ "expr": "sum(rate(polardbx_stats_tso_transaction_count_total{polardbx_name=\"$polardbx\", namespace=\"$namespace\"}[1m]))",
+ "hide": false,
+ "interval": "",
+ "legendFormat": "tso",
+ "queryType": "randomWalk",
+ "refId": "C"
+ }
+ ],
+ "title": "Transactions",
+ "type": "timeseries"
+ },
+ {
+ "datasource": {
+ "type": "prometheus",
+ "uid": "${datasource}"
+ },
+ "fieldConfig": {
+ "defaults": {
+ "color": {
+ "mode": "palette-classic"
+ },
+ "custom": {
+ "axisCenteredZero": false,
+ "axisColorMode": "text",
+ "axisLabel": "",
+ "axisPlacement": "auto",
+ "barAlignment": 0,
+ "drawStyle": "line",
+ "fillOpacity": 10,
+ "gradientMode": "opacity",
+ "hideFrom": {
+ "graph": false,
+ "legend": false,
+ "tooltip": false,
+ "viz": false
+ },
+ "lineInterpolation": "linear",
+ "lineStyle": {
+ "fill": "solid"
+ },
+ "lineWidth": 1,
+ "pointSize": 5,
+ "scaleDistribution": {
+ "type": "linear"
+ },
+ "showPoints": "never",
+ "spanNulls": true,
+ "stacking": {
+ "group": "A",
+ "mode": "none"
+ },
+ "thresholdsStyle": {
+ "mode": "off"
+ }
+ },
+ "mappings": [],
+ "thresholds": {
+ "mode": "absolute",
+ "steps": [
+ {
+ "color": "green"
+ }
+ ]
+ },
+ "unit": "µs"
+ },
+ "overrides": []
+ },
+ "gridPos": {
+ "h": 8,
+ "w": 6,
+ "x": 12,
+ "y": 53
+ },
+ "id": 33,
+ "options": {
+ "graph": {},
+ "legend": {
+ "calcs": [],
+ "displayMode": "list",
+ "placement": "bottom",
+ "showLegend": true
+ },
+ "tooltip": {
+ "mode": "multi",
+ "sort": "none"
+ }
+ },
+ "pluginVersion": "7.5.3",
+ "targets": [
+ {
+ "datasource": {
+ "uid": "${datasource}"
+ },
+ "editorMode": "code",
+ "exemplar": true,
+ "expr": "topk(5, sum by (schema) (\n label_replace(\n rate(polardbx_stc_request_time_cost_total{app_kubernetes_io_instance=\"$polardbx\", namespace=\"$namespace\", appname!~\"__.*__@.*|information_schema@.*|polardbx@.*\"}[1m]),\n \"schema\",\n \"$1\",\n \"appname\",\n \"(.*)@.*\"\n )\n )\n) by (schema)",
+ "interval": "",
+ "legendFormat": "{{ schema }}",
+ "queryType": "randomWalk",
+ "range": true,
+ "refId": "A"
+ }
+ ],
+ "title": "Processing Time (Top 5)",
+ "type": "timeseries"
+ },
+ {
+ "datasource": {
+ "type": "prometheus",
+ "uid": "${datasource}"
+ },
+ "fieldConfig": {
+ "defaults": {
+ "color": {
+ "mode": "palette-classic"
+ },
+ "custom": {
+ "axisCenteredZero": false,
+ "axisColorMode": "text",
+ "axisLabel": "",
+ "axisPlacement": "auto",
+ "barAlignment": 0,
+ "drawStyle": "line",
+ "fillOpacity": 10,
+ "gradientMode": "opacity",
+ "hideFrom": {
+ "graph": false,
+ "legend": false,
+ "tooltip": false,
+ "viz": false
+ },
+ "lineInterpolation": "linear",
+ "lineStyle": {
+ "fill": "solid"
+ },
+ "lineWidth": 1,
+ "pointSize": 5,
+ "scaleDistribution": {
+ "type": "linear"
+ },
+ "showPoints": "never",
+ "spanNulls": true,
+ "stacking": {
+ "group": "A",
+ "mode": "none"
+ },
+ "thresholdsStyle": {
+ "mode": "off"
+ }
+ },
+ "mappings": [],
+ "thresholds": {
+ "mode": "absolute",
+ "steps": [
+ {
+ "color": "green"
+ }
+ ]
+ },
+ "unit": "short"
+ },
+ "overrides": []
+ },
+ "gridPos": {
+ "h": 8,
+ "w": 6,
+ "x": 18,
+ "y": 53
+ },
+ "id": 34,
+ "options": {
+ "graph": {},
+ "legend": {
+ "calcs": [],
+ "displayMode": "list",
+ "placement": "bottom",
+ "showLegend": true
+ },
+ "tooltip": {
+ "mode": "multi",
+ "sort": "none"
+ }
+ },
+ "pluginVersion": "7.5.3",
+ "targets": [
+ {
+ "datasource": {
+ "uid": "${datasource}"
+ },
+ "editorMode": "code",
+ "exemplar": true,
+ "expr": "topk(5, sum by (schema) (\n label_replace(\n rate(polardbx_stc_active_physical_connection_count{app_kubernetes_io_instance=\"$polardbx\", namespace=\"$namespace\", appname!~\"__.*__@.*|information_schema@.*|polardbx@.*\"}[1m]),\n \"schema\",\n \"$1\",\n \"appname\",\n \"(.*)@.*\"\n )\n )\n) by (schema)",
+ "interval": "",
+ "legendFormat": "active - {{ schema }}",
+ "queryType": "randomWalk",
+ "range": true,
+ "refId": "A"
+ },
+ {
+ "datasource": {
+ "uid": "${datasource}"
+ },
+ "exemplar": true,
+ "expr": "topk(5, sum by (schema) (\n label_replace(\n rate(polardbx_stc_pooling_physical_connection_count{polardbx_name=\"$polardbx\", namespace=\"$namespace\", appname!~\"__.*__@.*|information_schema@.*|polardbx@.*\"}[1m]),\n \"schema\",\n \"$1\",\n \"appname\",\n \"(.*)@.*\"\n )\n )\n) by (schema)",
+ "hide": false,
+ "interval": "",
+ "legendFormat": "pooling - {{ schema }}",
+ "refId": "B"
+ }
+ ],
+ "title": "Physical Connection (Top 5)",
+ "type": "timeseries"
+ },
+ {
+ "collapsed": false,
+ "datasource": {
+ "type": "prometheus",
+ "uid": "prometheus"
+ },
+ "gridPos": {
+ "h": 1,
+ "w": 24,
+ "x": 0,
+ "y": 61
+ },
+ "id": 6,
+ "panels": [],
+ "targets": [
+ {
+ "datasource": {
+ "type": "prometheus",
+ "uid": "prometheus"
+ },
+ "refId": "A"
+ }
+ ],
+ "title": "Data Node",
+ "type": "row"
+ },
+ {
+ "datasource": {
+ "type": "prometheus",
+ "uid": "${datasource}"
+ },
+ "fieldConfig": {
+ "defaults": {
+ "color": {
+ "mode": "palette-classic"
+ },
+ "custom": {
+ "axisCenteredZero": false,
+ "axisColorMode": "text",
+ "axisLabel": "",
+ "axisPlacement": "auto",
+ "barAlignment": 0,
+ "drawStyle": "line",
+ "fillOpacity": 10,
+ "gradientMode": "opacity",
+ "hideFrom": {
+ "graph": false,
+ "legend": false,
+ "tooltip": false,
+ "viz": false
+ },
+ "lineInterpolation": "linear",
+ "lineStyle": {
+ "fill": "solid"
+ },
+ "lineWidth": 1,
+ "pointSize": 5,
+ "scaleDistribution": {
+ "type": "linear"
+ },
+ "showPoints": "never",
+ "spanNulls": true,
+ "stacking": {
+ "group": "A",
+ "mode": "none"
+ },
+ "thresholdsStyle": {
+ "mode": "off"
+ }
+ },
+ "mappings": [],
+ "thresholds": {
+ "mode": "absolute",
+ "steps": [
+ {
+ "color": "green"
+ }
+ ]
+ },
+ "unit": "short"
+ },
+ "overrides": []
+ },
+ "gridPos": {
+ "h": 8,
+ "w": 6,
+ "x": 0,
+ "y": 62
+ },
+ "id": 55,
+ "options": {
+ "graph": {},
+ "legend": {
+ "calcs": [],
+ "displayMode": "list",
+ "placement": "bottom",
+ "showLegend": true
+ },
+ "tooltip": {
+ "mode": "single",
+ "sort": "none"
+ }
+ },
+ "pluginVersion": "7.5.3",
+ "targets": [
+ {
+ "datasource": {
+ "uid": "${datasource}"
+ },
+ "editorMode": "code",
+ "exemplar": true,
+ "expr": "sum(rate(mysql_global_status_queries{app_kubernetes_io_instance=\"$polardbx\", app_kubernetes_io_component=\"dn\", namespace=\"$namespace\"}[1m]))",
+ "interval": "",
+ "legendFormat": "qps (physical)",
+ "queryType": "randomWalk",
+ "range": true,
+ "refId": "A"
+ }
+ ],
+ "title": "QPS",
+ "type": "timeseries"
+ },
+ {
+ "datasource": {
+ "type": "prometheus",
+ "uid": "${datasource}"
+ },
+ "fieldConfig": {
+ "defaults": {
+ "color": {
+ "mode": "palette-classic"
+ },
+ "custom": {
+ "axisCenteredZero": false,
+ "axisColorMode": "text",
+ "axisLabel": "",
+ "axisPlacement": "auto",
+ "barAlignment": 0,
+ "drawStyle": "line",
+ "fillOpacity": 10,
+ "gradientMode": "opacity",
+ "hideFrom": {
+ "graph": false,
+ "legend": false,
+ "tooltip": false,
+ "viz": false
+ },
+ "lineInterpolation": "linear",
+ "lineStyle": {
+ "fill": "solid"
+ },
+ "lineWidth": 1,
+ "pointSize": 5,
+ "scaleDistribution": {
+ "type": "linear"
+ },
+ "showPoints": "never",
+ "spanNulls": true,
+ "stacking": {
+ "group": "A",
+ "mode": "none"
+ },
+ "thresholdsStyle": {
+ "mode": "off"
+ }
+ },
+ "mappings": [],
+ "thresholds": {
+ "mode": "absolute",
+ "steps": [
+ {
+ "color": "green"
+ }
+ ]
+ },
+ "unit": "short"
+ },
+ "overrides": []
+ },
+ "gridPos": {
+ "h": 8,
+ "w": 6,
+ "x": 6,
+ "y": 62
+ },
+ "id": 56,
+ "options": {
+ "graph": {},
+ "legend": {
+ "calcs": [],
+ "displayMode": "list",
+ "placement": "bottom",
+ "showLegend": true
+ },
+ "tooltip": {
+ "mode": "multi",
+ "sort": "none"
+ }
+ },
+ "pluginVersion": "7.5.3",
+ "targets": [
+ {
+ "datasource": {
+ "uid": "${datasource}"
+ },
+ "editorMode": "code",
+ "exemplar": true,
+ "expr": "sum(mysql_global_status_threads_connected{app_kubernetes_io_instance=\"$polardbx\", app_kubernetes_io_component=\"dn\", namespace=\"$namespace\"})",
+ "interval": "",
+ "legendFormat": "connections",
+ "queryType": "randomWalk",
+ "range": true,
+ "refId": "A"
+ },
+ {
+ "datasource": {
+ "uid": "${datasource}"
+ },
+ "editorMode": "code",
+ "exemplar": true,
+ "expr": "sum(mysql_global_status_threads_running{app_kubernetes_io_instance=\"$polardbx\", app_kubernetes_io_component=\"dn\", namespace=\"$namespace\"})",
+ "hide": false,
+ "interval": "",
+ "legendFormat": "threads",
+ "range": true,
+ "refId": "B"
+ }
+ ],
+ "title": "Connections/Threads",
+ "type": "timeseries"
+ },
+ {
+ "datasource": {
+ "type": "prometheus",
+ "uid": "${datasource}"
+ },
+ "fieldConfig": {
+ "defaults": {
+ "color": {
+ "mode": "palette-classic"
+ },
+ "custom": {
+ "axisCenteredZero": false,
+ "axisColorMode": "text",
+ "axisLabel": "",
+ "axisPlacement": "auto",
+ "barAlignment": 0,
+ "drawStyle": "line",
+ "fillOpacity": 10,
+ "gradientMode": "opacity",
+ "hideFrom": {
+ "graph": false,
+ "legend": false,
+ "tooltip": false,
+ "viz": false
+ },
+ "lineInterpolation": "linear",
+ "lineStyle": {
+ "fill": "solid"
+ },
+ "lineWidth": 1,
+ "pointSize": 5,
+ "scaleDistribution": {
+ "type": "linear"
+ },
+ "showPoints": "never",
+ "spanNulls": true,
+ "stacking": {
+ "group": "A",
+ "mode": "none"
+ },
+ "thresholdsStyle": {
+ "mode": "off"
+ }
+ },
+ "mappings": [],
+ "thresholds": {
+ "mode": "absolute",
+ "steps": [
+ {
+ "color": "green"
+ }
+ ]
+ },
+ "unit": "bytes"
+ },
+ "overrides": []
+ },
+ "gridPos": {
+ "h": 8,
+ "w": 6,
+ "x": 12,
+ "y": 62
+ },
+ "id": 57,
+ "options": {
+ "graph": {},
+ "legend": {
+ "calcs": [],
+ "displayMode": "list",
+ "placement": "bottom",
+ "showLegend": true
+ },
+ "tooltip": {
+ "mode": "single",
+ "sort": "none"
+ }
+ },
+ "pluginVersion": "7.5.3",
+ "targets": [
+ {
+ "datasource": {
+ "uid": "${datasource}"
+ },
+ "editorMode": "code",
+ "exemplar": true,
+ "expr": "sum(mysql_global_status_innodb_buffer_pool_bytes_data{app_kubernetes_io_instance=\"$polardbx\", app_kubernetes_io_component=\"dn\", namespace=\"$namespace\"})",
+ "interval": "",
+ "legendFormat": "buffer pool size",
+ "queryType": "randomWalk",
+ "range": true,
+ "refId": "A"
+ }
+ ],
+ "title": "Buffer Pool Size (Total)",
+ "type": "timeseries"
+ },
+ {
+ "datasource": {
+ "type": "prometheus",
+ "uid": "${datasource}"
+ },
+ "fieldConfig": {
+ "defaults": {
+ "color": {
+ "mode": "palette-classic"
+ },
+ "custom": {
+ "axisCenteredZero": false,
+ "axisColorMode": "text",
+ "axisLabel": "",
+ "axisPlacement": "auto",
+ "barAlignment": 0,
+ "drawStyle": "line",
+ "fillOpacity": 10,
+ "gradientMode": "opacity",
+ "hideFrom": {
+ "graph": false,
+ "legend": false,
+ "tooltip": false,
+ "viz": false
+ },
+ "lineInterpolation": "linear",
+ "lineStyle": {
+ "fill": "solid"
+ },
+ "lineWidth": 1,
+ "pointSize": 5,
+ "scaleDistribution": {
+ "type": "linear"
+ },
+ "showPoints": "never",
+ "spanNulls": true,
+ "stacking": {
+ "group": "A",
+ "mode": "none"
+ },
+ "thresholdsStyle": {
+ "mode": "off"
+ }
+ },
+ "mappings": [],
+ "thresholds": {
+ "mode": "absolute",
+ "steps": [
+ {
+ "color": "green"
+ }
+ ]
+ },
+ "unit": "bytes"
+ },
+ "overrides": []
+ },
+ "gridPos": {
+ "h": 8,
+ "w": 6,
+ "x": 18,
+ "y": 62
+ },
+ "id": 58,
+ "options": {
+ "graph": {},
+ "legend": {
+ "calcs": [],
+ "displayMode": "list",
+ "placement": "bottom",
+ "showLegend": true
+ },
+ "tooltip": {
+ "mode": "multi",
+ "sort": "none"
+ }
+ },
+ "pluginVersion": "7.5.3",
+ "targets": [
+ {
+ "datasource": {
+ "uid": "${datasource}"
+ },
+ "editorMode": "code",
+ "exemplar": true,
+ "expr": "sum(mysql_global_status_innodb_buffer_pool_bytes_data{app_kubernetes_io_instance=\"$polardbx\", app_kubernetes_io_component=\"dn\", namespace=\"$namespace\"}) by (xstore_name)",
+ "interval": "",
+ "legendFormat": "{{ xstore_name }}",
+ "queryType": "randomWalk",
+ "range": true,
+ "refId": "A"
+ }
+ ],
+ "title": "Buffer Pool Size (DN)",
+ "type": "timeseries"
+ },
+ {
+ "datasource": {
+ "type": "prometheus",
+ "uid": "${datasource}"
+ },
+ "fieldConfig": {
+ "defaults": {
+ "color": {
+ "mode": "palette-classic"
+ },
+ "custom": {
+ "axisCenteredZero": false,
+ "axisColorMode": "text",
+ "axisLabel": "",
+ "axisPlacement": "auto",
+ "barAlignment": 0,
+ "drawStyle": "line",
+ "fillOpacity": 10,
+ "gradientMode": "opacity",
+ "hideFrom": {
+ "graph": false,
+ "legend": false,
+ "tooltip": false,
+ "viz": false
+ },
+ "lineInterpolation": "linear",
+ "lineStyle": {
+ "fill": "solid"
+ },
+ "lineWidth": 1,
+ "pointSize": 5,
+ "scaleDistribution": {
+ "log": 10,
+ "type": "log"
+ },
+ "showPoints": "never",
+ "spanNulls": true,
+ "stacking": {
+ "group": "A",
+ "mode": "none"
+ },
+ "thresholdsStyle": {
+ "mode": "off"
+ }
+ },
+ "mappings": [],
+ "thresholds": {
+ "mode": "absolute",
+ "steps": [
+ {
+ "color": "green"
+ }
+ ]
+ },
+ "unit": "short"
+ },
+ "overrides": []
+ },
+ "gridPos": {
+ "h": 8,
+ "w": 6,
+ "x": 0,
+ "y": 70
+ },
+ "id": 59,
+ "options": {
+ "graph": {},
+ "legend": {
+ "calcs": [],
+ "displayMode": "list",
+ "placement": "bottom",
+ "showLegend": true
+ },
+ "tooltip": {
+ "mode": "multi",
+ "sort": "none"
+ }
+ },
+ "pluginVersion": "7.5.3",
+ "targets": [
+ {
+ "datasource": {
+ "uid": "${datasource}"
+ },
+ "editorMode": "code",
+ "exemplar": true,
+ "expr": "sum(mysql_global_status_buffer_pool_dirty_pages{app_kubernetes_io_instance=\"$polardbx\", app_kubernetes_io_component=\"dn\", namespace=\"$namespace\"})",
+ "interval": "",
+ "legendFormat": "dirty",
+ "queryType": "randomWalk",
+ "range": true,
+ "refId": "A"
+ },
+ {
+ "datasource": {
+ "uid": "${datasource}"
+ },
+ "editorMode": "code",
+ "exemplar": true,
+ "expr": "sum(rate(mysql_global_status_buffer_pool_page_changes_total{app_kubernetes_io_instance=\"$polardbx\", app_kubernetes_io_component=\"dn\", namespace=\"$namespace\"}[10m]))",
+ "hide": false,
+ "interval": "",
+ "legendFormat": "flush",
+ "queryType": "randomWalk",
+ "range": true,
+ "refId": "B"
+ }
+ ],
+ "title": "Buffer Pool Dirty/Flush (Total)",
+ "type": "timeseries"
+ },
+ {
+ "datasource": {
+ "type": "prometheus",
+ "uid": "${datasource}"
+ },
+ "fieldConfig": {
+ "defaults": {
+ "color": {
+ "mode": "palette-classic"
+ },
+ "custom": {
+ "axisCenteredZero": false,
+ "axisColorMode": "text",
+ "axisLabel": "",
+ "axisPlacement": "auto",
+ "barAlignment": 0,
+ "drawStyle": "line",
+ "fillOpacity": 10,
+ "gradientMode": "opacity",
+ "hideFrom": {
+ "graph": false,
+ "legend": false,
+ "tooltip": false,
+ "viz": false
+ },
+ "lineInterpolation": "linear",
+ "lineStyle": {
+ "fill": "solid"
+ },
+ "lineWidth": 1,
+ "pointSize": 5,
+ "scaleDistribution": {
+ "type": "linear"
+ },
+ "showPoints": "never",
+ "spanNulls": true,
+ "stacking": {
+ "group": "A",
+ "mode": "none"
+ },
+ "thresholdsStyle": {
+ "mode": "off"
+ }
+ },
+ "mappings": [],
+ "thresholds": {
+ "mode": "absolute",
+ "steps": [
+ {
+ "color": "green"
+ }
+ ]
+ },
+ "unit": "short"
+ },
+ "overrides": []
+ },
+ "gridPos": {
+ "h": 8,
+ "w": 6,
+ "x": 6,
+ "y": 70
+ },
+ "id": 60,
+ "options": {
+ "graph": {},
+ "legend": {
+ "calcs": [],
+ "displayMode": "list",
+ "placement": "bottom",
+ "showLegend": true
+ },
+ "tooltip": {
+ "mode": "single",
+ "sort": "none"
+ }
+ },
+ "pluginVersion": "7.5.3",
+ "targets": [
+ {
+ "datasource": {
+ "uid": "${datasource}"
+ },
+ "editorMode": "code",
+ "exemplar": true,
+ "expr": "sum(mysql_global_status_innodb_row_lock_current_waits{app_kubernetes_io_instance=\"$polardbx\", app_kubernetes_io_component=\"dn\", namespace=\"$namespace\"})",
+ "interval": "",
+ "legendFormat": "current wait",
+ "queryType": "randomWalk",
+ "range": true,
+ "refId": "A"
+ }
+ ],
+ "title": "Row Lock Current Wait (Total)",
+ "type": "timeseries"
+ },
+ {
+ "datasource": {
+ "type": "prometheus",
+ "uid": "${datasource}"
+ },
+ "fieldConfig": {
+ "defaults": {
+ "color": {
+ "mode": "palette-classic"
+ },
+ "custom": {
+ "axisCenteredZero": false,
+ "axisColorMode": "text",
+ "axisLabel": "",
+ "axisPlacement": "auto",
+ "barAlignment": 0,
+ "drawStyle": "line",
+ "fillOpacity": 10,
+ "gradientMode": "opacity",
+ "hideFrom": {
+ "graph": false,
+ "legend": false,
+ "tooltip": false,
+ "viz": false
+ },
+ "lineInterpolation": "linear",
+ "lineStyle": {
+ "fill": "solid"
+ },
+ "lineWidth": 1,
+ "pointSize": 5,
+ "scaleDistribution": {
+ "type": "linear"
+ },
+ "showPoints": "never",
+ "spanNulls": true,
+ "stacking": {
+ "group": "A",
+ "mode": "none"
+ },
+ "thresholdsStyle": {
+ "mode": "off"
+ }
+ },
+ "mappings": [],
+ "thresholds": {
+ "mode": "absolute",
+ "steps": [
+ {
+ "color": "green"
+ }
+ ]
+ },
+ "unit": "short"
+ },
+ "overrides": []
+ },
+ "gridPos": {
+ "h": 8,
+ "w": 6,
+ "x": 12,
+ "y": 70
+ },
+ "id": 61,
+ "options": {
+ "graph": {},
+ "legend": {
+ "calcs": [],
+ "displayMode": "list",
+ "placement": "bottom",
+ "showLegend": true
+ },
+ "tooltip": {
+ "mode": "multi",
+ "sort": "none"
+ }
+ },
+ "pluginVersion": "7.5.3",
+ "targets": [
+ {
+ "datasource": {
+ "uid": "${datasource}"
+ },
+ "editorMode": "code",
+ "exemplar": true,
+ "expr": "sum(rate(mysql_global_status_innodb_buffer_pool_read_requests{app_kubernetes_io_instance=\"$polardbx\", app_kubernetes_io_component=\"dn\", namespace=\"$namespace\"}[1m]))",
+ "interval": "",
+ "legendFormat": "reads",
+ "queryType": "randomWalk",
+ "range": true,
+ "refId": "A"
+ },
+ {
+ "datasource": {
+ "uid": "${datasource}"
+ },
+ "editorMode": "code",
+ "exemplar": true,
+ "expr": "sum(rate(mysql_global_status_innodb_buffer_pool_write_requests{app_kubernetes_io_instance=\"$polardbx\", app_kubernetes_io_component=\"dn\", namespace=\"$namespace\"}[1m]))",
+ "hide": false,
+ "interval": "",
+ "legendFormat": "writes",
+ "queryType": "randomWalk",
+ "range": true,
+ "refId": "B"
+ }
+ ],
+ "title": "Logical Reads/Writes (Total)",
+ "type": "timeseries"
+ },
+ {
+ "datasource": {
+ "type": "prometheus",
+ "uid": "${datasource}"
+ },
+ "fieldConfig": {
+ "defaults": {
+ "color": {
+ "mode": "palette-classic"
+ },
+ "custom": {
+ "axisCenteredZero": false,
+ "axisColorMode": "text",
+ "axisLabel": "",
+ "axisPlacement": "auto",
+ "barAlignment": 0,
+ "drawStyle": "line",
+ "fillOpacity": 10,
+ "gradientMode": "opacity",
+ "hideFrom": {
+ "graph": false,
+ "legend": false,
+ "tooltip": false,
+ "viz": false
+ },
+ "lineInterpolation": "linear",
+ "lineStyle": {
+ "fill": "solid"
+ },
+ "lineWidth": 1,
+ "pointSize": 5,
+ "scaleDistribution": {
+ "type": "linear"
+ },
+ "showPoints": "never",
+ "spanNulls": true,
+ "stacking": {
+ "group": "A",
+ "mode": "none"
+ },
+ "thresholdsStyle": {
+ "mode": "off"
+ }
+ },
+ "mappings": [],
+ "thresholds": {
+ "mode": "absolute",
+ "steps": [
+ {
+ "color": "green"
+ }
+ ]
+ },
+ "unit": "short"
+ },
+ "overrides": []
+ },
+ "gridPos": {
+ "h": 8,
+ "w": 6,
+ "x": 18,
+ "y": 70
+ },
+ "id": 62,
+ "options": {
+ "graph": {},
+ "legend": {
+ "calcs": [],
+ "displayMode": "list",
+ "placement": "bottom",
+ "showLegend": true
+ },
+ "tooltip": {
+ "mode": "multi",
+ "sort": "none"
+ }
+ },
+ "pluginVersion": "7.5.3",
+ "targets": [
+ {
+ "datasource": {
+ "uid": "${datasource}"
+ },
+ "editorMode": "code",
+ "exemplar": true,
+ "expr": "sum(rate(mysql_global_status_innodb_log_writes{app_kubernetes_io_instance=\"$polardbx\", app_kubernetes_io_component=\"dn\", namespace=\"$namespace\"}[1m]))",
+ "interval": "",
+ "legendFormat": "log writes",
+ "queryType": "randomWalk",
+ "range": true,
+ "refId": "A"
+ },
+ {
+ "datasource": {
+ "uid": "${datasource}"
+ },
+ "editorMode": "code",
+ "exemplar": true,
+ "expr": "sum(rate(mysql_global_status_innodb_data_reads{app_kubernetes_io_instance=\"$polardbx\", app_kubernetes_io_component=\"dn\", namespace=\"$namespace\"}[1m]))",
+ "hide": false,
+ "interval": "",
+ "legendFormat": "data reads",
+ "queryType": "randomWalk",
+ "range": true,
+ "refId": "B"
+ },
+ {
+ "datasource": {
+ "uid": "${datasource}"
+ },
+ "exemplar": true,
+ "expr": "sum(rate(mysql_global_status_innodb_data_fsyncs{polardbx_name=\"$polardbx\", polardbx_role=\"dn\", xstore_role=\"leader\", namespace=\"$namespace\"}[1m]))",
+ "hide": false,
+ "interval": "",
+ "legendFormat": "data fsyncs",
+ "queryType": "randomWalk",
+ "refId": "C"
+ },
+ {
+ "datasource": {
+ "uid": "${datasource}"
+ },
+ "exemplar": true,
+ "expr": "sum(rate(mysql_global_status_innodb_os_log_fsyncs{polardbx_name=\"$polardbx\", polardbx_role=\"dn\", xstore_role=\"leader\", namespace=\"$namespace\"}[1m]))",
+ "hide": false,
+ "interval": "",
+ "legendFormat": "log fsyncs",
+ "queryType": "randomWalk",
+ "refId": "D"
+ }
+ ],
+ "title": "IOPS (Total)",
+ "type": "timeseries"
+ },
+ {
+ "datasource": {
+ "type": "prometheus",
+ "uid": "${datasource}"
+ },
+ "fieldConfig": {
+ "defaults": {
+ "color": {
+ "mode": "palette-classic"
+ },
+ "custom": {
+ "axisCenteredZero": false,
+ "axisColorMode": "text",
+ "axisLabel": "",
+ "axisPlacement": "auto",
+ "barAlignment": 0,
+ "drawStyle": "line",
+ "fillOpacity": 10,
+ "gradientMode": "opacity",
+ "hideFrom": {
+ "graph": false,
+ "legend": false,
+ "tooltip": false,
+ "viz": false
+ },
+ "lineInterpolation": "linear",
+ "lineStyle": {
+ "fill": "solid"
+ },
+ "lineWidth": 1,
+ "pointSize": 5,
+ "scaleDistribution": {
+ "type": "linear"
+ },
+ "showPoints": "never",
+ "spanNulls": true,
+ "stacking": {
+ "group": "A",
+ "mode": "none"
+ },
+ "thresholdsStyle": {
+ "mode": "off"
+ }
+ },
+ "mappings": [],
+ "thresholds": {
+ "mode": "absolute",
+ "steps": [
+ {
+ "color": "green"
+ }
+ ]
+ },
+ "unit": "bytes"
+ },
+ "overrides": []
+ },
+ "gridPos": {
+ "h": 8,
+ "w": 6,
+ "x": 0,
+ "y": 78
+ },
+ "id": 63,
+ "options": {
+ "graph": {},
+ "legend": {
+ "calcs": [],
+ "displayMode": "list",
+ "placement": "bottom",
+ "showLegend": true
+ },
+ "tooltip": {
+ "mode": "multi",
+ "sort": "none"
+ }
+ },
+ "pluginVersion": "7.5.3",
+ "targets": [
+ {
+ "datasource": {
+ "uid": "${datasource}"
+ },
+ "editorMode": "code",
+ "exemplar": true,
+ "expr": "sum(rate(mysql_global_status_innodb_data_written{app_kubernetes_io_instance=\"$polardbx\", app_kubernetes_io_component=\"dn\", namespace=\"$namespace\"}[1m]))",
+ "interval": "",
+ "legendFormat": "data written/s",
+ "queryType": "randomWalk",
+ "range": true,
+ "refId": "A"
+ },
+ {
+ "datasource": {
+ "uid": "${datasource}"
+ },
+ "editorMode": "code",
+ "exemplar": true,
+ "expr": "sum(rate(mysql_global_status_innodb_data_read{app_kubernetes_io_instance=\"$polardbx\", app_kubernetes_io_component=\"dn\", namespace=\"$namespace\"}[1m]))",
+ "hide": false,
+ "interval": "",
+ "legendFormat": "data read/s",
+ "queryType": "randomWalk",
+ "range": true,
+ "refId": "B"
+ },
+ {
+ "datasource": {
+ "uid": "${datasource}"
+ },
+ "exemplar": true,
+ "expr": "sum(rate(mysql_global_status_innodb_os_log_written{polardbx_name=\"$polardbx\", polardbx_role=\"dn\", xstore_role=\"leader\", namespace=\"$namespace\"}[1m]))",
+ "hide": false,
+ "interval": "",
+ "legendFormat": "log writtern/s",
+ "queryType": "randomWalk",
+ "refId": "C"
+ }
+ ],
+ "title": "IO Throughput (Total)",
+ "type": "timeseries"
+ },
+ {
+ "datasource": {
+ "type": "prometheus",
+ "uid": "${datasource}"
+ },
+ "fieldConfig": {
+ "defaults": {
+ "color": {
+ "mode": "palette-classic"
+ },
+ "custom": {
+ "axisCenteredZero": false,
+ "axisColorMode": "text",
+ "axisLabel": "",
+ "axisPlacement": "auto",
+ "barAlignment": 0,
+ "drawStyle": "line",
+ "fillOpacity": 10,
+ "gradientMode": "opacity",
+ "hideFrom": {
+ "graph": false,
+ "legend": false,
+ "tooltip": false,
+ "viz": false
+ },
+ "lineInterpolation": "linear",
+ "lineStyle": {
+ "fill": "solid"
+ },
+ "lineWidth": 1,
+ "pointSize": 5,
+ "scaleDistribution": {
+ "type": "linear"
+ },
+ "showPoints": "never",
+ "spanNulls": true,
+ "stacking": {
+ "group": "A",
+ "mode": "none"
+ },
+ "thresholdsStyle": {
+ "mode": "off"
+ }
+ },
+ "mappings": [],
+ "thresholds": {
+ "mode": "absolute",
+ "steps": [
+ {
+ "color": "green"
+ }
+ ]
+ },
+ "unit": "bytes"
+ },
+ "overrides": []
+ },
+ "gridPos": {
+ "h": 8,
+ "w": 6,
+ "x": 6,
+ "y": 78
+ },
+ "id": 64,
+ "options": {
+ "graph": {},
+ "legend": {
+ "calcs": [],
+ "displayMode": "list",
+ "placement": "bottom",
+ "showLegend": true
+ },
+ "tooltip": {
+ "mode": "multi",
+ "sort": "none"
+ }
+ },
+ "pluginVersion": "7.5.3",
+ "targets": [
+ {
+ "datasource": {
+ "uid": "${datasource}"
+ },
+ "editorMode": "code",
+ "exemplar": true,
+ "expr": "sum(rate(mysql_global_status_bytes_received{app_kubernetes_io_instance=\"$polardbx\", app_kubernetes_io_component=\"dn\", namespace=\"$namespace\"}[1m]))",
+ "interval": "",
+ "legendFormat": "received/s",
+ "queryType": "randomWalk",
+ "range": true,
+ "refId": "A"
+ },
+ {
+ "datasource": {
+ "uid": "${datasource}"
+ },
+ "editorMode": "code",
+ "exemplar": true,
+ "expr": "sum(rate(mysql_global_status_bytes_sent{app_kubernetes_io_instance=\"$polardbx\", app_kubernetes_io_component=\"dn\", namespace=\"$namespace\"}[1m]))",
+ "hide": false,
+ "interval": "",
+ "legendFormat": "sent/s",
+ "queryType": "randomWalk",
+ "range": true,
+ "refId": "B"
+ }
+ ],
+ "title": "Network (Total)",
+ "type": "timeseries"
+ },
+ {
+ "datasource": {
+ "type": "prometheus",
+ "uid": "${datasource}"
+ },
+ "fieldConfig": {
+ "defaults": {
+ "color": {
+ "mode": "palette-classic"
+ },
+ "custom": {
+ "axisCenteredZero": false,
+ "axisColorMode": "text",
+ "axisLabel": "",
+ "axisPlacement": "auto",
+ "barAlignment": 0,
+ "drawStyle": "line",
+ "fillOpacity": 10,
+ "gradientMode": "opacity",
+ "hideFrom": {
+ "graph": false,
+ "legend": false,
+ "tooltip": false,
+ "viz": false
+ },
+ "lineInterpolation": "linear",
+ "lineStyle": {
+ "fill": "solid"
+ },
+ "lineWidth": 1,
+ "pointSize": 5,
+ "scaleDistribution": {
+ "type": "linear"
+ },
+ "showPoints": "never",
+ "spanNulls": true,
+ "stacking": {
+ "group": "A",
+ "mode": "none"
+ },
+ "thresholdsStyle": {
+ "mode": "off"
+ }
+ },
+ "mappings": [],
+ "thresholds": {
+ "mode": "absolute",
+ "steps": [
+ {
+ "color": "green"
+ }
+ ]
+ },
+ "unit": "short"
+ },
+ "overrides": []
+ },
+ "gridPos": {
+ "h": 8,
+ "w": 6,
+ "x": 12,
+ "y": 78
+ },
+ "id": 65,
+ "options": {
+ "graph": {},
+ "legend": {
+ "calcs": [],
+ "displayMode": "list",
+ "placement": "bottom",
+ "showLegend": true
+ },
+ "tooltip": {
+ "mode": "multi",
+ "sort": "none"
+ }
+ },
+ "pluginVersion": "7.5.3",
+ "targets": [
+ {
+ "datasource": {
+ "uid": "${datasource}"
+ },
+ "editorMode": "code",
+ "exemplar": true,
+ "expr": "sum(rate(mysql_global_status_commands_total{polapp_kubernetes_io_instanceardbx_name=\"$polardbx\", app_kubernetes_io_component=\"dn\", namespace=\"$namespace\", command=~\"xa_.*\"}[1m])) by (command)",
+ "interval": "",
+ "legendFormat": "{{ command }}",
+ "queryType": "randomWalk",
+ "range": true,
+ "refId": "A"
+ }
+ ],
+ "title": "XA Transactions (Total)",
+ "type": "timeseries"
+ },
+ {
+ "datasource": {
+ "type": "prometheus",
+ "uid": "${datasource}"
+ },
+ "fieldConfig": {
+ "defaults": {
+ "color": {
+ "mode": "palette-classic"
+ },
+ "custom": {
+ "axisCenteredZero": false,
+ "axisColorMode": "text",
+ "axisLabel": "",
+ "axisPlacement": "auto",
+ "barAlignment": 0,
+ "drawStyle": "line",
+ "fillOpacity": 10,
+ "gradientMode": "none",
+ "hideFrom": {
+ "graph": false,
+ "legend": false,
+ "tooltip": false,
+ "viz": false
+ },
+ "lineInterpolation": "linear",
+ "lineWidth": 1,
+ "pointSize": 5,
+ "scaleDistribution": {
+ "type": "linear"
+ },
+ "showPoints": "never",
+ "spanNulls": true,
+ "stacking": {
+ "group": "A",
+ "mode": "none"
+ },
+ "thresholdsStyle": {
+ "mode": "off"
+ }
+ },
+ "mappings": [],
+ "thresholds": {
+ "mode": "absolute",
+ "steps": [
+ {
+ "color": "green"
+ },
+ {
+ "color": "red",
+ "value": 80
+ }
+ ]
+ },
+ "unit": "short"
+ },
+ "overrides": []
+ },
+ "gridPos": {
+ "h": 8,
+ "w": 6,
+ "x": 18,
+ "y": 78
+ },
+ "id": 66,
+ "options": {
+ "graph": {},
+ "legend": {
+ "calcs": [],
+ "displayMode": "list",
+ "placement": "bottom",
+ "showLegend": true
+ },
+ "tooltip": {
+ "mode": "multi",
+ "sort": "none"
+ }
+ },
+ "pluginVersion": "7.5.3",
+ "targets": [
+ {
+ "datasource": {
+ "uid": "${datasource}"
+ },
+ "editorMode": "code",
+ "exemplar": true,
+ "expr": "sum(rate(mysql_global_status_commands_total{app_kubernetes_io_instance=\"$polardbx\", app_kubernetes_io_component=\"dn\", namespace=\"$namespace\", command=~\"begin|commit|rollback\"}[1m])) by (command)",
+ "interval": "",
+ "legendFormat": "{{ command }}",
+ "queryType": "randomWalk",
+ "range": true,
+ "refId": "A"
+ }
+ ],
+ "title": "Transactions (Total)",
+ "type": "timeseries"
+ },
+ {
+ "collapsed": false,
+ "datasource": {
+ "type": "prometheus",
+ "uid": "prometheus"
+ },
+ "gridPos": {
+ "h": 1,
+ "w": 24,
+ "x": 0,
+ "y": 86
+ },
+ "id": 68,
+ "panels": [],
+ "targets": [
+ {
+ "datasource": {
+ "type": "prometheus",
+ "uid": "prometheus"
+ },
+ "refId": "A"
+ }
+ ],
+ "title": "CDC Node",
+ "type": "row"
+ },
+ {
+ "datasource": {
+ "type": "prometheus",
+ "uid": "${datasource}"
+ },
+ "fieldConfig": {
+ "defaults": {
+ "color": {
+ "mode": "palette-classic"
+ },
+ "custom": {
+ "axisCenteredZero": false,
+ "axisColorMode": "text",
+ "axisLabel": "",
+ "axisPlacement": "auto",
+ "barAlignment": 0,
+ "drawStyle": "line",
+ "fillOpacity": 10,
+ "gradientMode": "opacity",
+ "hideFrom": {
+ "graph": false,
+ "legend": false,
+ "tooltip": false,
+ "viz": false
+ },
+ "lineInterpolation": "linear",
+ "lineStyle": {
+ "fill": "solid"
+ },
+ "lineWidth": 1,
+ "pointSize": 5,
+ "scaleDistribution": {
+ "type": "linear"
+ },
+ "showPoints": "never",
+ "spanNulls": true,
+ "stacking": {
+ "group": "A",
+ "mode": "none"
+ },
+ "thresholdsStyle": {
+ "mode": "off"
+ }
+ },
+ "mappings": [],
+ "thresholds": {
+ "mode": "absolute",
+ "steps": [
+ {
+ "color": "green"
+ },
+ {
+ "color": "red",
+ "value": 80
+ }
+ ]
+ },
+ "unit": "ms"
+ },
+ "overrides": []
+ },
+ "gridPos": {
+ "h": 8,
+ "w": 6,
+ "x": 0,
+ "y": 87
+ },
+ "id": 76,
+ "options": {
+ "graph": {},
+ "legend": {
+ "calcs": [],
+ "displayMode": "list",
+ "placement": "bottom",
+ "showLegend": true
+ },
+ "tooltip": {
+ "mode": "single",
+ "sort": "none"
+ }
+ },
+ "pluginVersion": "7.5.3",
+ "targets": [
+ {
+ "datasource": {
+ "uid": "${datasource}"
+ },
+ "editorMode": "code",
+ "exemplar": true,
+ "expr": "polardbx_cdc_dumper_delay_in_millisecond{namespace=\"$namespace\"}\n* on(namespace,pod)\ngroup_left() polardbx_cdc_up{app_kubernetes_io_instance=\"$polardbx\", namespace=\"$namespace\", app_kubernetes_io_component=\"cdc\"}",
+ "interval": "",
+ "legendFormat": "{{ pod }}",
+ "queryType": "randomWalk",
+ "range": true,
+ "refId": "A"
+ }
+ ],
+ "title": "Delay(ms)",
+ "type": "timeseries"
+ },
+ {
+ "datasource": {
+ "type": "prometheus",
+ "uid": "${datasource}"
+ },
+ "fieldConfig": {
+ "defaults": {
+ "color": {
+ "mode": "palette-classic"
+ },
+ "custom": {
+ "axisCenteredZero": false,
+ "axisColorMode": "text",
+ "axisLabel": "",
+ "axisPlacement": "auto",
+ "barAlignment": 0,
+ "drawStyle": "line",
+ "fillOpacity": 10,
+ "gradientMode": "opacity",
+ "hideFrom": {
+ "graph": false,
+ "legend": false,
+ "tooltip": false,
+ "viz": false
+ },
+ "lineInterpolation": "linear",
+ "lineStyle": {
+ "fill": "solid"
+ },
+ "lineWidth": 1,
+ "pointSize": 5,
+ "scaleDistribution": {
+ "type": "linear"
+ },
+ "showPoints": "never",
+ "spanNulls": true,
+ "stacking": {
+ "group": "A",
+ "mode": "none"
+ },
+ "thresholdsStyle": {
+ "mode": "off"
+ }
+ },
+ "mappings": [],
+ "thresholds": {
+ "mode": "absolute",
+ "steps": [
+ {
+ "color": "green"
+ },
+ {
+ "color": "red",
+ "value": 80
+ }
+ ]
+ },
+ "unit": "percent"
+ },
+ "overrides": []
+ },
+ "gridPos": {
+ "h": 8,
+ "w": 6,
+ "x": 6,
+ "y": 87
+ },
+ "id": 77,
+ "options": {
+ "graph": {},
+ "legend": {
+ "calcs": [],
+ "displayMode": "list",
+ "placement": "bottom",
+ "showLegend": true
+ },
+ "tooltip": {
+ "mode": "single",
+ "sort": "none"
+ }
+ },
+ "pluginVersion": "7.5.3",
+ "targets": [
+ {
+ "datasource": {
+ "uid": "${datasource}"
+ },
+ "editorMode": "code",
+ "exemplar": true,
+ "expr": "polardbx_cdc_dumper_jvm_heap_usage{namespace=\"$namespace\"}\n* on(namespace,pod)\ngroup_left() polardbx_cdc_up{app_kubernetes_io_instance=\"$polardbx\", namespace=\"$namespace\", app_kubernetes_io_component=\"cdc\"}",
+ "interval": "",
+ "legendFormat": "{{ pod }}",
+ "queryType": "randomWalk",
+ "range": true,
+ "refId": "A"
+ }
+ ],
+ "title": "Dumper Heap Usage",
+ "type": "timeseries"
+ }
+ ],
+ "refresh": "10s",
+ "schemaVersion": 37,
+ "style": "dark",
+ "tags": [],
+ "templating": {
+ "list": [
+ {
+ "current": {
+ "isNone": true,
+ "selected": false,
+ "text": "None",
+ "value": ""
+ },
+ "datasource": {
+ "type": "prometheus",
+ "uid": "${datasource}"
+ },
+ "definition": "label_values(kube_pod_info, cluster)",
+ "hide": 2,
+ "includeAll": false,
+ "multi": false,
+ "name": "cluster",
+ "options": [],
+ "query": {
+ "query": "label_values(kube_pod_info, cluster)",
+ "refId": "StandardVariableQuery"
+ },
+ "refresh": 1,
+ "regex": "",
+ "skipUrlSync": false,
+ "sort": 0,
+ "tagValuesQuery": "",
+ "tagsQuery": "",
+ "type": "query",
+ "useTags": false
+ },
+ {
+ "current": {
+ "selected": false,
+ "text": "default",
+ "value": "default"
+ },
+ "hide": 0,
+ "includeAll": false,
+ "multi": false,
+ "name": "datasource",
+ "options": [],
+ "query": "prometheus",
+ "queryValue": "",
+ "refresh": 1,
+ "regex": "",
+ "skipUrlSync": false,
+ "type": "datasource"
+ },
+ {
+ "current": {
+ "selected": false,
+ "text": "default",
+ "value": "default"
+ },
+ "datasource": {
+ "type": "prometheus",
+ "uid": "${datasource}"
+ },
+ "definition": "label_values(polardbx_up, namespace)",
+ "hide": 0,
+ "includeAll": false,
+ "multi": false,
+ "name": "namespace",
+ "options": [],
+ "query": {
+ "query": "label_values(polardbx_up, namespace)",
+ "refId": "StandardVariableQuery"
+ },
+ "refresh": 1,
+ "regex": "/^(?!kube\\-system|.*\\-operator\\-system|monitoring|loki)/",
+ "skipUrlSync": false,
+ "sort": 0,
+ "tagValuesQuery": "",
+ "tagsQuery": "",
+ "type": "query",
+ "useTags": false
+ },
+ {
+ "current": {
+ "selected": false,
+ "text": "pxc-v4",
+ "value": "pxc-v4"
+ },
+ "datasource": {
+ "type": "prometheus",
+ "uid": "${datasource}"
+ },
+ "definition": "label_values(polardbx_up{namespace=\"$namespace\"}, app_kubernetes_io_instance)",
+ "hide": 0,
+ "includeAll": false,
+ "multi": false,
+ "name": "polardbx",
+ "options": [],
+ "query": {
+ "query": "label_values(polardbx_up{namespace=\"$namespace\"}, app_kubernetes_io_instance)",
+ "refId": "StandardVariableQuery"
+ },
+ "refresh": 1,
+ "regex": "",
+ "skipUrlSync": false,
+ "sort": 1,
+ "tagValuesQuery": "",
+ "tagsQuery": "",
+ "type": "query",
+ "useTags": false
+ }
+ ]
+ },
+ "time": {
+ "from": "now-5m",
+ "to": "now"
+ },
+ "timepicker": {},
+ "timezone": "",
+ "title": "PolarDB-X Overview",
+ "uid": "EByXvnuGk",
+ "version": 2,
+ "weekStart": ""
+}
\ No newline at end of file
diff --git a/deploy/polardbx/scripts/gms-init.sql b/deploy/polardbx/scripts/gms-init.sql
new file mode 100644
index 00000000000..496edb02233
--- /dev/null
+++ b/deploy/polardbx/scripts/gms-init.sql
@@ -0,0 +1,153 @@
+CREATE DATABASE IF NOT EXISTS polardbx_meta_db;
+USE polardbx_meta_db;
+
+CREATE TABLE IF NOT EXISTS server_info (
+ id BIGINT(11) NOT NULL auto_increment,
+ gmt_created TIMESTAMP NOT NULL DEFAULT CURRENT_TIMESTAMP,
+ gmt_modified TIMESTAMP NOT NULL DEFAULT CURRENT_TIMESTAMP on UPDATE CURRENT_TIMESTAMP,
+ inst_id VARCHAR(128) NOT NULL,
+ inst_type INT(11) NOT NULL,
+ ip VARCHAR(128) NOT NULL,
+ port INT(11) NOT NULL,
+ htap_port INT(11) NOT NULL,
+ mgr_port INT(11) NOT NULL,
+ mpp_port INT(11) NOT NULL,
+ status INT(11) NOT NULL,
+ region_id VARCHAR(128) DEFAULT NULL,
+ azone_id VARCHAR(128) DEFAULT NULL,
+ idc_id VARCHAR(128) DEFAULT NULL,
+ cpu_core INT(11) DEFAULT NULL,
+ mem_size INT(11) DEFAULT NULL,
+ extras text DEFAULT NULL,
+ PRIMARY KEY (id),
+ UNIQUE KEY uk_inst_id_addr (inst_id, ip, port),
+ INDEX idx_inst_id_status (inst_id, status)
+) engine = innodb DEFAULT charset = utf8;
+
+CREATE TABLE IF NOT EXISTS storage_info (
+ id BIGINT(11) NOT NULL auto_increment,
+ gmt_created TIMESTAMP NOT NULL DEFAULT CURRENT_TIMESTAMP,
+ gmt_modified TIMESTAMP NOT NULL DEFAULT CURRENT_TIMESTAMP on UPDATE CURRENT_TIMESTAMP,
+ inst_id VARCHAR(128) NOT NULL,
+ storage_inst_id VARCHAR(128) NOT NULL,
+ storage_master_inst_id VARCHAR(128) NOT NULL,
+ ip VARCHAR(128) NOT NULL,
+ port INT(11) NOT NULL comment 'port for mysql',
+ xport INT(11) DEFAULT NULL comment 'port for x-protocol',
+ user VARCHAR(128) NOT NULL,
+ passwd_enc text NOT NULL,
+ storage_type INT(11) NOT NULL comment '0:x-cluster, 1:mysql, 2:polardb',
+ inst_kind INT(11) NOT NULL comment '0:master, 1:slave, 2:metadb',
+ status INT(11) NOT NULL comment '0:storage ready, 1:storage not_ready',
+ region_id VARCHAR(128) DEFAULT NULL,
+ azone_id VARCHAR(128) DEFAULT NULL,
+ idc_id VARCHAR(128) DEFAULT NULL,
+ max_conn INT(11) NOT NULL,
+ cpu_core INT(11) DEFAULT NULL,
+ mem_size INT(11) DEFAULT NULL comment 'mem unit: MB',
+ is_vip INT(11) DEFAULT NULL COMMENT '0:ip is NOT vip, 1:ip is vip',
+ extras text DEFAULT NULL COMMENT 'reserve for extra info',
+ PRIMARY KEY (id),
+ INDEX idx_inst_id_status (inst_id, status),
+ UNIQUE KEY uk_inst_id_addr (storage_inst_id, ip, port, inst_kind)
+) engine = innodb DEFAULT charset = utf8;
+
+CREATE TABLE if not exists user_priv (
+ id bigint(11) NOT NULL AUTO_INCREMENT,
+ gmt_created timestamp NOT NULL DEFAULT CURRENT_TIMESTAMP,
+ gmt_modified timestamp NOT NULL DEFAULT CURRENT_TIMESTAMP ON UPDATE CURRENT_TIMESTAMP,
+ user_name char(32) COLLATE utf8_unicode_ci NOT NULL DEFAULT '',
+ host char(60) COLLATE utf8_unicode_ci NOT NULL DEFAULT '',
+ password char(100) COLLATE utf8_unicode_ci NOT NULL,
+ select_priv tinyint(1) NOT NULL DEFAULT '0',
+ insert_priv tinyint(1) NOT NULL DEFAULT '0',
+ update_priv tinyint(1) NOT NULL DEFAULT '0',
+ delete_priv tinyint(1) NOT NULL DEFAULT '0',
+ create_priv tinyint(1) NOT NULL DEFAULT '0',
+ drop_priv tinyint(1) NOT NULL DEFAULT '0',
+ grant_priv tinyint(1) NOT NULL DEFAULT '0',
+ index_priv tinyint(1) NOT NULL DEFAULT '0',
+ alter_priv tinyint(1) NOT NULL DEFAULT '0',
+ show_view_priv int(11) NOT NULL DEFAULT '0',
+ create_view_priv int(11) NOT NULL DEFAULT '0',
+ create_user_priv int(11) NOT NULL DEFAULT '0',
+ meta_db_priv int(11) NOT NULL DEFAULT '0',
+ PRIMARY KEY (id),
+ UNIQUE KEY uk (user_name, host)
+) ENGINE = InnoDB DEFAULT CHARSET = utf8 COLLATE = utf8_unicode_ci COMMENT = 'Users and global privileges';
+
+CREATE TABLE IF NOT EXISTS quarantine_config (
+ id BIGINT(11) NOT NULL auto_increment,
+ gmt_created TIMESTAMP NOT NULL DEFAULT CURRENT_TIMESTAMP,
+ gmt_modified TIMESTAMP NOT NULL DEFAULT CURRENT_TIMESTAMP on UPDATE CURRENT_TIMESTAMP,
+ inst_id VARCHAR(100) CHARACTER SET utf8 COLLATE utf8_unicode_ci NOT NULL,
+ group_name VARCHAR(200) CHARACTER SET utf8 COLLATE utf8_unicode_ci NOT NULL,
+ net_work_type VARCHAR(100) CHARACTER SET utf8 COLLATE utf8_unicode_ci DEFAULT NULL,
+ security_ip_type VARCHAR(100) CHARACTER SET utf8 COLLATE utf8_unicode_ci DEFAULT NULL,
+ security_ips text CHARACTER SET utf8 COLLATE utf8_unicode_ci,
+ PRIMARY KEY (id),
+ UNIQUE KEY uk (inst_id, group_name)
+) engine = innodb DEFAULT charset = utf8 comment = 'Quarantine config';
+
+
+CREATE TABLE IF NOT EXISTS config_listener (
+ id bigint(11) NOT NULL AUTO_INCREMENT,
+ gmt_created TIMESTAMP NOT NULL DEFAULT CURRENT_TIMESTAMP,
+ gmt_modified TIMESTAMP NOT NULL DEFAULT CURRENT_TIMESTAMP ON UPDATE CURRENT_TIMESTAMP,
+ data_id varchar(200) NOT NULL,
+ status int NOT NULL COMMENT '0:normal, 1:removed',
+ op_version bigint NOT NULL,
+ extras varchar(1024) DEFAULT NULL,
+ PRIMARY KEY (id),
+ INDEX idx_modify_ts (gmt_modified),
+ INDEX idx_status (status),
+ UNIQUE KEY uk_data_id (data_id)
+) ENGINE = InnoDB DEFAULT CHARSET = utf8;
+
+create table if not exists inst_config (
+ id bigint(11) NOT NULL AUTO_INCREMENT,
+ gmt_created timestamp NOT NULL DEFAULT CURRENT_TIMESTAMP,
+ gmt_modified timestamp NOT NULL DEFAULT CURRENT_TIMESTAMP ON UPDATE CURRENT_TIMESTAMP,
+ inst_id varchar(128) NOT NULL,
+ param_key varchar(128) NOT NULL,
+ param_val varchar(1024) NOT NULL,
+ PRIMARY KEY (id),
+ UNIQUE KEY uk_inst_id_key (inst_id, param_key)
+) ENGINE = InnoDB DEFAULT CHARSET = utf8;
+
+CREATE TABLE IF NOT EXISTS polardbx_extra (
+ id BIGINT(11) NOT NULL auto_increment,
+ inst_id VARCHAR(128) NOT NULL,
+ name VARCHAR(128) NOT NULL,
+ type VARCHAR(10) NOT NULL,
+ comment VARCHAR(256) NOT NULL,
+ status INT(4) NOT NULL,
+ gmt_created TIMESTAMP NOT NULL DEFAULT CURRENT_TIMESTAMP,
+ gmt_modified TIMESTAMP NOT NULL DEFAULT CURRENT_TIMESTAMP on
+ UPDATE CURRENT_TIMESTAMP,
+ PRIMARY KEY (id),
+ UNIQUE uk_inst_id_name_type (inst_id, name, type)
+) engine = innodb DEFAULT charset = utf8 COLLATE = utf8_unicode_ci comment = 'extra table for polardbx manager';
+
+CREATE TABLE IF NOT EXISTS schema_change (
+ id BIGINT(11) NOT NULL AUTO_INCREMENT,
+ table_name varchar(64) NOT NULL,
+ version int unsigned NOT NULL,
+ gmt_created timestamp NOT NULL DEFAULT CURRENT_TIMESTAMP,
+ gmt_modified timestamp NOT NULL DEFAULT CURRENT_TIMESTAMP ON UPDATE CURRENT_TIMESTAMP,
+ PRIMARY KEY (id),
+ UNIQUE KEY table_name (table_name)
+ ) ENGINE = innodb DEFAULT CHARSET=utf8;
+
+CREATE TABLE IF NOT EXISTS k8s_topology (
+ id BIGINT(11) NOT NULL AUTO_INCREMENT,
+ uid VARCHAR(128) NOT NULL,
+ name VARCHAR(128) NOT NULL,
+ type VARCHAR(10) NOT NULL,
+ gmt_created TIMESTAMP NOT NULL DEFAULT CURRENT_TIMESTAMP,
+ gmt_modified TIMESTAMP NOT NULL DEFAULT CURRENT_TIMESTAMP ON UPDATE CURRENT_TIMESTAMP,
+ PRIMARY KEY (id),
+ UNIQUE KEY(uid),
+ UNIQUE KEY(name, type)
+) ENGINE = InnoDB DEFAULT CHARSET = utf8 COLLATE = utf8_unicode_ci COMMENT = 'PolarDBX K8s Topology';
+
diff --git a/deploy/polardbx/scripts/gms-metadata.tpl b/deploy/polardbx/scripts/gms-metadata.tpl
new file mode 100644
index 00000000000..ca9c9cec2c2
--- /dev/null
+++ b/deploy/polardbx/scripts/gms-metadata.tpl
@@ -0,0 +1,23 @@
+USE polardbx_meta_db;
+
+INSERT IGNORE INTO schema_change(table_name, version) VALUES('user_priv', 1);
+INSERT IGNORE INTO quarantine_config VALUES (NULL, NOW(), NOW(), '$KB_CLUSTER_NAME', 'default', NULL, NULL, '0.0.0.0/0');
+
+INSERT IGNORE INTO config_listener (id, gmt_created, gmt_modified, data_id, status, op_version, extras) VALUES (NULL, NOW(), NOW(), 'polardbx.server.info.$KB_CLUSTER_NAME', 0, 0, NULL);
+INSERT IGNORE INTO config_listener (id, gmt_created, gmt_modified, data_id, status, op_version, extras) VALUES (NULL, NOW(), NOW(), 'polardbx.storage.info.$KB_CLUSTER_NAME', 0, 0, NULL);
+INSERT IGNORE INTO config_listener (id, gmt_created, gmt_modified, data_id, status, op_version, extras) VALUES (NULL, NOW(), NOW(), 'polardbx.inst.config.$KB_CLUSTER_NAME', 0, 0, NULL);
+INSERT IGNORE INTO config_listener (id, gmt_created, gmt_modified, data_id, status, op_version, extras) VALUES (NULL, NOW(), NOW(), 'polardbx.quarantine.config.$KB_CLUSTER_NAME', 0, 0, NULL);
+INSERT IGNORE INTO config_listener (id, gmt_created, gmt_modified, data_id, status, op_version, extras) VALUES (NULL, NOW(), NOW(), 'polardbx.privilege.info', 0, 0, NULL);
+
+INSERT IGNORE INTO inst_config (inst_id, param_key,param_val) values ('$KB_CLUSTER_NAME','CONN_POOL_XPROTO_META_DB_PORT','0');
+INSERT IGNORE INTO inst_config (inst_id, param_key,param_val) values ('$KB_CLUSTER_NAME','CDC_STARTUP_MODE','1');
+INSERT IGNORE INTO inst_config (inst_id, param_key,param_val) values ('$KB_CLUSTER_NAME','CONN_POOL_MAX_POOL_SIZE','500');
+INSERT IGNORE INTO inst_config (inst_id, param_key,param_val) values ('$KB_CLUSTER_NAME','MAX_PREPARED_STMT_COUNT','500000');
+
+INSERT IGNORE INTO storage_info (id, gmt_created, gmt_modified, inst_id, storage_inst_id, storage_master_inst_id,ip, port, xport, user, passwd_enc, storage_type, inst_kind, status, region_id, azone_id, idc_id, max_conn, cpu_core, mem_size, is_vip, extras)
+ VALUES (NULL, NOW(), NOW(), '$KB_CLUSTER_NAME', '$GMS_SVC_NAME', '$GMS_SVC_NAME', '$GMS_HOST', '3306', '31600', '$metaDbUser', '$ENC_PASSWORD', '3', '2', '0', NULL, NULL, NULL, 10000, 4, 34359738368 , '0', '');
+
+INSERT IGNORE INTO user_priv (id, gmt_created, gmt_modified, user_name, host, password, select_priv, insert_priv, update_priv, delete_priv, create_priv, drop_priv, grant_priv, index_priv, alter_priv, show_view_priv, create_view_priv, create_user_priv, meta_db_priv)
+ VALUES (NULL, now(), now(), '$metaDbUser', '%', '$SHA1_ENC_PASSWORD', '1', '1', '1', '1', '1', '1', '1', '1', '1', '1', '1', '1', '1');
+
+UPDATE config_listener SET op_version = op_version + 1 WHERE data_id = 'polardbx.privilege.info';
\ No newline at end of file
diff --git a/deploy/polardbx/scripts/metadb-setup.tpl b/deploy/polardbx/scripts/metadb-setup.tpl
new file mode 100644
index 00000000000..6b8e8dd4b5e
--- /dev/null
+++ b/deploy/polardbx/scripts/metadb-setup.tpl
@@ -0,0 +1,37 @@
+#!/bin/sh
+
+until mysql -h$GMS_SVC_NAME -P$GMS_SVC_PORT -u$metaDbUser -p$metaDbNonEncPasswd -e 'select 1'; do
+ sleep 1;
+ echo "wait gms ready"
+done
+
+function generate_dn_init_sql() {
+ echo "$DN_HEADLESS_SVC_NAME" | tr ',' '\n' | while IFS= read -r item
+ do
+ DN_HOSTNAME=$item
+ DN_NAME=$(echo "$DN_HOSTNAME" | cut -d'.' -f2 | sed s/-headless//)
+ dn_init_sql="INSERT IGNORE INTO storage_info (id, gmt_created, gmt_modified, inst_id, storage_inst_id, storage_master_inst_id,ip, port, xport, user, passwd_enc, storage_type, inst_kind, status, region_id, azone_id, idc_id, max_conn, cpu_core, mem_size, is_vip, extras)
+ VALUES (NULL, NOW(), NOW(), '$KB_CLUSTER_NAME', '$DN_NAME', '$DN_NAME', '$DN_HOSTNAME', '3306', '31600', '$metaDbUser', '$ENC_PASSWORD', '3', '0', '0', NULL, NULL, NULL, 10000, 4, 34359738368 , '0', '');"
+ echo $dn_init_sql >> /scripts/gms-init-metadata.sql
+ done
+ echo "UPDATE config_listener SET op_version = op_version + 1 WHERE data_id = 'polardbx.storage.info.$KB_CLUSTER_NAME'" >> /scripts/gms-init-metadata.sql
+}
+
+ENC_PASSWORD=$(echo -n "$metaDbNonEncPasswd" | openssl enc -aes-128-ecb -K "$(printf "%s" "$dnPasswordKey" | od -An -tx1 | tr -d " \n")" -base64)
+SHA1_ENC_PASSWORD=$(echo -n "$metaDbNonEncPasswd" | sha1sum | cut -d ' ' -f1)
+echo "export metaDbPasswd=$ENC_PASSWORD" >> /shared/env.sh
+
+SOURCE_CMD="mysql -h$GMS_SVC_NAME -P$GMS_SVC_PORT -u$metaDbUser -p$metaDbNonEncPasswd -e 'source /scripts/gms-init.sql'"
+eval $SOURCE_CMD
+
+GMS_HOST=$GMS_SVC_NAME"."$KB_NAMESPACE".svc.cluster.local"
+
+eval "gms_metadata_sql=\"$(cat /scripts/gms-metadata.tpl)\""
+
+echo $gms_metadata_sql > /scripts/gms-init-metadata.sql
+generate_dn_init_sql
+
+cat /scripts/gms-init-metadata.sql
+
+eval "mysql -h$GMS_SVC_NAME -P$GMS_SVC_PORT -u$metaDbUser -p$metaDbNonEncPasswd -e 'source /scripts/gms-init-metadata.sql'"
+
diff --git a/deploy/polardbx/scripts/xstore-post-start.tpl b/deploy/polardbx/scripts/xstore-post-start.tpl
new file mode 100644
index 00000000000..e11f09673da
--- /dev/null
+++ b/deploy/polardbx/scripts/xstore-post-start.tpl
@@ -0,0 +1,24 @@
+#!/bin/sh
+# usage: xstore-post-start.sh type_name
+# type_name: component.type, in uppercase.
+
+TYPE_NAME=$1
+
+# setup shared-channel.json
+SHARED_CHANNEL_JSON='{"nodes": ['
+
+i=0
+while [ $i -lt $(eval echo \$KB_"$TYPE_NAME"_N) ]; do
+ hostname=$(eval echo \$KB_"$TYPE_NAME"_"$i"_HOSTNAME)
+ pod=$(echo "$hostname" | cut -d'.' -f1)
+
+ NODE_OBJECT=$(printf '{"pod": "%s", "host": "%s", "port": 11306, "role": "candidate", "node_name": "%s" }' "$pod" "$hostname" "$pod")
+ SHARED_CHANNEL_JSON+="$NODE_OBJECT,"
+ i=$(( i + 1))
+done
+
+SHARED_CHANNEL_JSON=${SHARED_CHANNEL_JSON%,}
+SHARED_CHANNEL_JSON+=']}'
+
+mkdir -p /data/shared/
+echo $SHARED_CHANNEL_JSON > /data/shared/shared-channel.json
diff --git a/deploy/polardbx/scripts/xstore-setup.tpl b/deploy/polardbx/scripts/xstore-setup.tpl
new file mode 100644
index 00000000000..2dcbf872e5f
--- /dev/null
+++ b/deploy/polardbx/scripts/xstore-setup.tpl
@@ -0,0 +1,17 @@
+#!/bin/sh
+# usage: xstore-setup.sh
+# setup root account for xstore and run entrypoint
+
+function setup_account() {
+ until myc -e 'select 1'; do
+ sleep 1;
+ echo "wait mysql ready"
+ done
+ echo "mysql is ok"
+ myc -e "SET sql_log_bin=OFF;SET force_revise=ON;CREATE USER IF NOT EXISTS $KB_SERVICE_USER IDENTIFIED BY '$KB_SERVICE_PASSWORD';GRANT ALL PRIVILEGES ON *.* TO $KB_SERVICE_USER;ALTER USER $KB_SERVICE_USER IDENTIFIED BY '$KB_SERVICE_PASSWORD';"
+
+}
+
+
+setup_account &
+/tools/xstore/current/venv/bin/python3 /tools/xstore/current/entrypoint.py
diff --git a/deploy/polardbx/templates/NOTES.txt b/deploy/polardbx/templates/NOTES.txt
new file mode 100644
index 00000000000..3b27cadf8f2
--- /dev/null
+++ b/deploy/polardbx/templates/NOTES.txt
@@ -0,0 +1,14 @@
+Thanks for installing PolarDB-X using KubeBlocks!
+
+1. Run the following command to create your first PolarDB-X cluster:
+
+```
+kbcli cluster create pxc --cluster-definition polardbx
+```
+
+2. Port-forward service to localhost and connect to PolarDB-X cluster:
+
+```
+kubectl port-forward svc/pxc-cn 3306:3306
+mysql -h127.0.0.1 -upolardbx_root
+```
\ No newline at end of file
diff --git a/deploy/polardbx/templates/_helpers.tpl b/deploy/polardbx/templates/_helpers.tpl
new file mode 100644
index 00000000000..963bdc3f623
--- /dev/null
+++ b/deploy/polardbx/templates/_helpers.tpl
@@ -0,0 +1,62 @@
+{{/*
+Expand the name of the chart.
+*/}}
+{{- define "polardbx.name" -}}
+{{- default .Chart.Name .Values.nameOverride | trunc 63 | trimSuffix "-" }}
+{{- end }}
+
+{{/*
+Create a default fully qualified app name.
+We truncate at 63 chars because some Kubernetes name fields are limited to this (by the DNS naming spec).
+If release name contains chart name it will be used as a full name.
+*/}}
+{{- define "polardbx.fullname" -}}
+{{- if .Values.fullnameOverride }}
+{{- .Values.fullnameOverride | trunc 63 | trimSuffix "-" }}
+{{- else }}
+{{- $name := default .Chart.Name .Values.nameOverride }}
+{{- if contains $name .Release.Name }}
+{{- .Release.Name | trunc 63 | trimSuffix "-" }}
+{{- else }}
+{{- printf "%s-%s" .Release.Name $name | trunc 63 | trimSuffix "-" }}
+{{- end }}
+{{- end }}
+{{- end }}
+
+{{/*
+Create chart name and version as used by the chart label.
+*/}}
+{{- define "polardbx.chart" -}}
+{{- printf "%s-%s" .Chart.Name .Chart.Version | replace "+" "_" | trunc 63 | trimSuffix "-" }}
+{{- end }}
+
+{{/*
+Common labels
+*/}}
+{{- define "polardbx.labels" -}}
+helm.sh/chart: {{ include "polardbx.chart" . }}
+{{ include "polardbx.selectorLabels" . }}
+{{- if .Chart.AppVersion }}
+app.kubernetes.io/version: {{ .Chart.AppVersion | quote }}
+{{- end }}
+app.kubernetes.io/managed-by: {{ .Release.Service }}
+{{- end }}
+
+{{/*
+Selector labels
+*/}}
+{{- define "polardbx.selectorLabels" -}}
+app.kubernetes.io/name: {{ include "polardbx.name" . }}
+app.kubernetes.io/instance: {{ .Release.Name }}
+{{- end }}
+
+{{/*
+Create the name of the service account to use
+*/}}
+{{- define "polardbx.serviceAccountName" -}}
+{{- if .Values.serviceAccount.create }}
+{{- default (include "polardbx.fullname" .) .Values.serviceAccount.name }}
+{{- else }}
+{{- default "default" .Values.serviceAccount.name }}
+{{- end }}
+{{- end }}
diff --git a/deploy/polardbx/templates/clusterDefintion.yaml b/deploy/polardbx/templates/clusterDefintion.yaml
new file mode 100644
index 00000000000..42a8bc7c70d
--- /dev/null
+++ b/deploy/polardbx/templates/clusterDefintion.yaml
@@ -0,0 +1,728 @@
+apiVersion: apps.kubeblocks.io/v1alpha1
+kind: ClusterDefinition
+metadata:
+ name: polardbx
+ labels:
+ {{- include "polardbx.labels" . | nindent 4 }}
+spec:
+ connectionCredential:
+ username: "polardbx_root"
+ password: "$(RANDOM_PASSWD)"
+ endpoint: "$(SVC_FQDN):$(SVC_PORT_polardbx)"
+ host: "$(SVC_FQDN)"
+ port: "$(SVC_PORT_polardbx)"
+ metaDbPasswd: "$(RANDOM_PASSWD)"
+ componentDefs:
+ - name: gms
+ scriptSpecs:
+ - name: polardbx-scripts
+ templateRef: polardbx-scripts
+ volumeName: scripts
+ namespace: {{ .Release.Namespace }}
+ defaultMode: 0555
+ workloadType: Consensus
+ characterType: polardbx
+ consensusSpec:
+ leader:
+ name: "leader"
+ accessMode: ReadWrite
+ followers:
+ - name: "follower"
+ accessMode: Readonly
+ rsmSpec:
+ roles:
+ - name: "leader"
+ accessMode: ReadWrite
+ isLeader: true
+ canVote: true
+ - name: "follower"
+ accessMode: Readonly
+ canVote: true
+ roleProbe:
+ roleUpdateMechanism: DirectAPIServerEventUpdate
+ probeActions:
+ - image: "arey/mysql-client:latest"
+ command:
+ - mysql
+ - "-h127.0.0.1"
+ - "-P3306"
+ - "-uroot"
+ - "-N"
+ - "-B"
+ - "-e"
+ - "\"select role from information_schema.alisql_cluster_local\""
+ - "|"
+ - "xargs echo -n"
+ service:
+ ports:
+ - name: mysql
+ port: 3306
+ targetPort: 3306
+ - name: metrics
+ port: 9104
+ targetPort: 9104
+ monitor:
+ builtIn: false
+ exporterConfig:
+ scrapePort: 9104
+ scrapePath: "/metrics"
+ podSpec:
+ volumes: &xstoreVolumes
+ - emptyDir: {}
+ name: xstore-tools
+ - downwardAPI:
+ defaultMode: 420
+ items:
+ - fieldRef:
+ apiVersion: v1
+ fieldPath: metadata.labels
+ path: labels
+ - fieldRef:
+ apiVersion: v1
+ fieldPath: metadata.annotations
+ path: annotations
+ - fieldRef:
+ apiVersion: v1
+ fieldPath: metadata.annotations['runmode']
+ path: runmode
+ - fieldRef:
+ apiVersion: v1
+ fieldPath: metadata.name
+ path: name
+ - fieldRef:
+ apiVersion: v1
+ fieldPath: metadata.namespace
+ path: namespace
+ name: podinfo
+ initContainers: &xsotreInitContainers
+ - name: tools-updater
+ command: ["/bin/ash"]
+ args: ["-c", "./hack/update.sh /target"]
+ env:
+ - name: NODE_NAME
+ valueFrom:
+ fieldRef:
+ apiVersion: v1
+ fieldPath: spec.nodeName
+ volumeMounts:
+ - name: xstore-tools
+ mountPath: /target
+ containers:
+ - name: engine
+ command: ["/scripts/xstore-setup.sh"]
+ lifecycle:
+ postStart:
+ exec:
+ command:
+ - /scripts/xstore-post-start.sh
+ - GMS
+ env: &xstoreEngineEnv
+ - name: LANG
+ value: en_US.utf8
+ - name: LC_ALL
+ value: en_US.utf8
+ - name: ENGINE
+ value: galaxy
+ - name: ENGINE_HOME
+ value: /opt/galaxy_engine
+ - name: NODE_ROLE
+ value: candidate
+ - name: NODE_IP
+ valueFrom:
+ fieldRef:
+ apiVersion: v1
+ fieldPath: status.hostIP
+ - name: NODE_NAME
+ valueFrom:
+ fieldRef:
+ apiVersion: v1
+ fieldPath: spec.nodeName
+ - name: POD_IP
+ valueFrom:
+ fieldRef:
+ apiVersion: v1
+ fieldPath: status.podIP
+ - name: POD_NAME
+ valueFrom:
+ fieldRef:
+ apiVersion: v1
+ fieldPath: metadata.name
+ - name: LIMITS_CPU
+ valueFrom:
+ resourceFieldRef:
+ containerName: engine
+ resource: limits.cpu
+ divisor: "1m"
+ - name: LIMITS_MEM
+ valueFrom:
+ resourceFieldRef:
+ containerName: engine
+ resource: limits.memory
+ - name: PORT_MYSQL
+ value: "3306"
+ - name: PORT_PAXOS
+ value: "11306"
+ - name: PORT_POLARX
+ value: "31600"
+ - name: KB_SERVICE_USER
+ value: "polardbx_root"
+ - name: KB_SERVICE_PASSWORD
+ valueFrom:
+ secretKeyRef:
+ name: $(CONN_CREDENTIAL_SECRET_NAME)
+ key: password
+ optional: false
+ - name: RSM_COMPATIBILITY_MODE
+ value: "true"
+ ports: &xstoreEnginePorts
+ - name: mysql
+ containerPort: 3306
+ - name: paxos
+ containerPort: 11306
+ - name: polarx
+ containerPort: 31600
+ startupProbe:
+ failureThreshold: 60
+ tcpSocket:
+ port: mysql
+ initialDelaySeconds: 5
+ periodSeconds: 10
+ successThreshold: 1
+ timeoutSeconds: 30
+ volumeMounts: &xstoreEngineVolumeMounts
+ - name: data
+ mountPath: /data/mysql
+ - name: data-log
+ mountPath: /data-log/mysql
+ - name: xstore-tools
+ mountPath: /tools/xstore
+ - name: scripts
+ mountPath: /scripts/xstore-post-start.sh
+ subPath: xstore-post-start.sh
+ - name: scripts
+ mountPath: /scripts/xstore-setup.sh
+ subPath: xstore-setup.sh
+ - name: podinfo
+ mountPath: /etc/podinfo
+ - name: exporter
+ imagePullPolicy: IfNotPresent
+ ports:
+ - name: metrics
+ containerPort: 9104
+ protocol: TCP
+ env:
+ - name: "MYSQL_MONITOR_USER"
+ valueFrom:
+ secretKeyRef:
+ name: $(CONN_CREDENTIAL_SECRET_NAME)
+ key: username
+ - name: "MYSQL_MONITOR_PASSWORD"
+ valueFrom:
+ secretKeyRef:
+ name: $(CONN_CREDENTIAL_SECRET_NAME)
+ key: password
+ - name: "DATA_SOURCE_NAME"
+ value: "$(MYSQL_MONITOR_USER):$(MYSQL_MONITOR_PASSWORD)@(localhost:3306)/"
+ - name: dn
+ scriptSpecs:
+ - name: polardbx-scripts
+ templateRef: polardbx-scripts
+ volumeName: scripts
+ namespace: {{ .Release.Namespace }}
+ defaultMode: 0555
+ workloadType: Consensus
+ characterType: polardbx
+ componentDefRef:
+ - &gmsRef
+ componentDefName: gms
+ componentRefEnv:
+ - name: GMS_SVC_PORT
+ valueFrom:
+ type: FieldRef
+ fieldPath: $.componentDef.service.ports[?(@.name == "mysql")].port
+ - name: GMS_SVC_NAME
+ valueFrom:
+ type: ServiceRef
+ consensusSpec:
+ leader:
+ name: "leader"
+ accessMode: ReadWrite
+ followers:
+ - name: "follower"
+ accessMode: Readonly
+ rsmSpec:
+ roles:
+ - name: "leader"
+ accessMode: ReadWrite
+ isLeader: true
+ canVote: true
+ - name: "follower"
+ accessMode: Readonly
+ canVote: true
+ roleProbe:
+ roleUpdateMechanism: DirectAPIServerEventUpdate
+ probeActions:
+ - image: "arey/mysql-client:latest"
+ command:
+ - mysql
+ - "-h127.0.0.1"
+ - "-P3306"
+ - "-uroot"
+ - "-N"
+ - "-B"
+ - "-e"
+ - "\"select role from information_schema.alisql_cluster_local\""
+ - "|"
+ - "xargs echo -n"
+ service:
+ ports:
+ - name: mysql
+ port: 3306
+ targetPort: 3306
+ monitor:
+ builtIn: false
+ exporterConfig:
+ scrapePort: 9104
+ scrapePath: "/metrics"
+ podSpec:
+ volumes: *xstoreVolumes
+ initContainers: *xsotreInitContainers
+ containers:
+ - name: engine
+ command: [ "/scripts/xstore-setup.sh" ]
+ lifecycle:
+ postStart:
+ exec:
+ command:
+ - /scripts/xstore-post-start.sh
+ - DN
+ env: *xstoreEngineEnv
+ ports: *xstoreEnginePorts
+ startupProbe:
+ failureThreshold: 60
+ tcpSocket:
+ port: mysql
+ initialDelaySeconds: 20
+ periodSeconds: 10
+ successThreshold: 1
+ timeoutSeconds: 30
+ volumeMounts: *xstoreEngineVolumeMounts
+ - name: exporter
+ imagePullPolicy: IfNotPresent
+ ports:
+ - name: metrics
+ containerPort: 9104
+ protocol: TCP
+ env:
+ - name: "MYSQL_MONITOR_USER"
+ valueFrom:
+ secretKeyRef:
+ name: $(CONN_CREDENTIAL_SECRET_NAME)
+ key: username
+ - name: "MYSQL_MONITOR_PASSWORD"
+ valueFrom:
+ secretKeyRef:
+ name: $(CONN_CREDENTIAL_SECRET_NAME)
+ key: password
+ - name: "DATA_SOURCE_NAME"
+ value: "$(MYSQL_MONITOR_USER):$(MYSQL_MONITOR_PASSWORD)@(localhost:3306)/"
+ - name: cn
+ scriptSpecs:
+ - name: polardbx-scripts
+ templateRef: polardbx-scripts
+ volumeName: scripts
+ namespace: {{ .Release.Namespace }}
+ defaultMode: 0555
+ workloadType: Stateless
+ characterType: mysql
+ componentDefRef:
+ - *gmsRef
+ - componentDefName: dn
+ componentRefEnv:
+ - name: DN_SVC_PORT
+ valueFrom:
+ type: FieldRef
+ fieldPath: $.componentDef.service.ports[?(@.name == "mysql")].port
+ - name: DN_HEADLESS_SVC_NAME
+ valueFrom:
+ type: HeadlessServiceRef
+ format: $(POD_FQDN){{ .Values.clusterDomain }}
+ joinWith: ","
+ service:
+ ports:
+ - name: mysql
+ port: 3306
+ targetPort: 3306
+ - name: metrics
+ port: 9104
+ targetPort: 9104
+ monitor:
+ builtIn: false
+ exporterConfig:
+ scrapePort: 9104
+ scrapePath: "/metrics"
+ podSpec:
+ shareProcessNamespace: true # For jmx collector
+ volumes:
+ - name: shared
+ emptyDir: {}
+ initContainers:
+ - name: metadb-init
+ command: ["/scripts/metadb-setup.sh"]
+ env:
+ - name: metaDbAddr
+ value: "$(GMS_SVC_NAME):$(GMS_SVC_PORT)"
+ - name: metaDbName
+ value: "polardbx_meta_db"
+ - name: metaDbUser
+ valueFrom:
+ secretKeyRef:
+ name: $(CONN_CREDENTIAL_SECRET_NAME)
+ key: username
+ optional: false
+ - name: metaDbNonEncPasswd
+ valueFrom:
+ secretKeyRef:
+ name: $(CONN_CREDENTIAL_SECRET_NAME)
+ key: password
+ optional: false
+ - name: dnPasswordKey
+ value: "$(metaDbNonEncPasswd)$(metaDbNonEncPasswd)"
+ - name: switchCloud
+ value: aliyun
+ - name: metaDbConn
+ value: "mysql -h$(GMS_SVC_NAME) -P3306 -u$(metaDbUser) -p$(metaDbNonEncPasswd) -D$(metaDbName)"
+ volumeMounts:
+ - name: scripts
+ mountPath: /scripts/metadb-setup.sh
+ subPath: metadb-setup.sh
+ - name: scripts
+ mountPath: /scripts/gms-init.sql
+ subPath: gms-init.sql
+ - name: scripts
+ mountPath: /scripts/gms-metadata.tpl
+ subPath: gms-metadata.tpl
+ - name: shared
+ mountPath: /shared
+ - name: init
+ command: [ "sh" ]
+ args: [ "-c", 'source /shared/env.sh && /polardbx-init' ]
+ env: &cnEngineEnv
+ - name: POD_ID
+ valueFrom:
+ fieldRef:
+ apiVersion: v1
+ fieldPath: metadata.name
+ - name: POD_IP
+ valueFrom:
+ fieldRef:
+ apiVersion: v1
+ fieldPath: status.podIP
+ - name: HOST_IP
+ valueFrom:
+ fieldRef:
+ apiVersion: v1
+ fieldPath: status.hostIP
+ - name: NODE_NAME
+ valueFrom:
+ fieldRef:
+ apiVersion: v1
+ fieldPath: spec.nodeName
+ - name: metaDbAddr
+ value: "$(GMS_SVC_NAME):$(GMS_SVC_PORT)"
+ - name: metaDbName
+ value: "polardbx_meta_db"
+ - name: metaDbUser
+ valueFrom:
+ secretKeyRef:
+ name: $(CONN_CREDENTIAL_SECRET_NAME)
+ key: username
+ optional: false
+ - name: metaDbNonEncPasswd
+ valueFrom:
+ secretKeyRef:
+ name: $(CONN_CREDENTIAL_SECRET_NAME)
+ key: password
+ optional: false
+ - name: switchCloud
+ value: aliyun
+ - name: metaDbConn
+ value: "mysql -h$(GMS_SVC_NAME) -P3306 -u$(metaDbUser) -p$(metaDbPasswd) -D$(metaDbName)"
+ - name: dnPasswordKey
+ value: "$(metaDbNonEncPasswd)$(metaDbNonEncPasswd)"
+ - name: metaDbXprotoPort
+ value: "0"
+ - name: storageDbXprotoPort
+ value: "0"
+ - name: instanceId
+ value: "$(KB_CLUSTER_NAME)"
+ - name: instanceType
+ value: "0"
+ - name: serverPort
+ value: "3306"
+ - name: mgrPort
+ value: "3406"
+ - name: mppPort
+ value: "3506"
+ - name: htapPort
+ value: "3606"
+ - name: logPort
+ value: "8507"
+ - name: ins_id
+ value: dummy
+ - name: polarx_dummy_log_port
+ value: "$(logPort)"
+ - name: polarx_dummy_ssh_port
+ value: "-1"
+ - name: cpuCore
+ valueFrom:
+ resourceFieldRef:
+ containerName: engine
+ resource: limits.cpu
+ - name: memSize
+ valueFrom:
+ resourceFieldRef:
+ containerName: engine
+ resource: limits.memory
+ - name: cpu_cores
+ valueFrom:
+ resourceFieldRef:
+ containerName: engine
+ resource: limits.cpu
+ - name: memory
+ valueFrom:
+ resourceFieldRef:
+ containerName: engine
+ resource: limits.memory
+ - name: galaxyXProtocol
+ value: "1"
+ - name: processorHandler
+ value: "1"
+ - name: processors
+ value: "1"
+ - name: serverExecutor
+ value: "1024"
+ - name: TDDL_OPTS
+ value: -Dpod.id=$(POD_ID) -XX:+UnlockExperimentalVMOptions -XX:+UseWisp2 -Dio.grpc.netty.shaded.io.netty.transport.noNative=true
+ -Dio.netty.transport.noNative=true -DinstanceVersion=8.0.3
+ volumeMounts:
+ - name: shared
+ mountPath: /shared
+ containers:
+ - name: engine
+ command:
+ - /bin/bash
+ - -c
+ args:
+ - "source /shared/env.sh && /home/admin/entrypoint.sh 20"
+ env: *cnEngineEnv
+ ports:
+ - containerPort: 3306
+ name: mysql
+ protocol: TCP
+ - containerPort: 3406
+ name: mgr
+ protocol: TCP
+ - containerPort: 3506
+ name: mpp
+ protocol: TCP
+ - containerPort: 3606
+ name: htap
+ protocol: TCP
+ - containerPort: 8507
+ name: log
+ protocol: TCP
+ startupProbe:
+ failureThreshold: 60
+ tcpSocket:
+ port: mysql
+ initialDelaySeconds: 5
+ periodSeconds: 10
+ successThreshold: 1
+ timeoutSeconds: 30
+ livenessProbe:
+ failureThreshold: 60
+ tcpSocket:
+ port: mysql
+ initialDelaySeconds: 5
+ periodSeconds: 10
+ successThreshold: 1
+ timeoutSeconds: 30
+ readinessProbe:
+ failureThreshold: 60
+ tcpSocket:
+ port: mysql
+ initialDelaySeconds: 5
+ periodSeconds: 10
+ successThreshold: 1
+ timeoutSeconds: 30
+ volumeMounts:
+ - name: tmp
+ mountPath: /tmp
+ - name: polardbx-log
+ mountPath: /home/admin/drds-server/logs
+ - name: polardbx-spill
+ mountPath: /home/admin/drds-server/spill
+ - name: shared
+ mountPath: /shared
+ - name: exporter
+ args:
+ - -collectors.process
+ - -collectors.jvm
+ - -target.type=CN
+ - -target.port=3406
+ - -web.listen-addr=:9104
+ - -web.metrics-path=/metrics
+ env:
+ - name: GOMAXPROCS
+ value: "1"
+ ports:
+ - containerPort: 9104
+ name: metrics
+ protocol: TCP
+ volumeMounts:
+ - name: tmp
+ mountPath: /tmp
+ - name: cdc
+ scriptSpecs:
+ - name: polardbx-scripts
+ templateRef: polardbx-scripts
+ volumeName: scripts
+ namespace: {{ .Release.Namespace }}
+ defaultMode: 0555
+ workloadType: Stateless
+ characterType: mysql
+ componentDefRef:
+ - *gmsRef
+ - componentDefName: cn
+ componentRefEnv:
+ - name: CN_SVC_PORT
+ valueFrom:
+ type: FieldRef
+ fieldPath: $.componentDef.service.ports[?(@.name == "mysql")].port
+ - name: CN_SVC_NAME
+ valueFrom:
+ type: ServiceRef
+ service:
+ ports:
+ - name: mysql
+ port: 3306
+ targetPort: 3306
+ - name: metrics
+ port: 9104
+ targetPort: 9104
+ monitor:
+ builtIn: false
+ exporterConfig:
+ scrapePort: 9104
+ scrapePath: "/metrics"
+ podSpec:
+ initContainers:
+ - name: wait-cn-ready
+ command:
+ - bin/sh
+ - -c
+ - |
+ until mysql -h$CN_SVC_NAME -P$CN_SVC_PORT -u$polarx_username -p$polarx_password -e 'select 1'; do
+ sleep 1;
+ echo "cn is not ready"
+ done
+ env:
+ - name: polarx_username
+ valueFrom:
+ secretKeyRef:
+ name: $(CONN_CREDENTIAL_SECRET_NAME)
+ key: username
+ optional: false
+ - name: polarx_password
+ valueFrom:
+ secretKeyRef:
+ name: $(CONN_CREDENTIAL_SECRET_NAME)
+ key: password
+ optional: false
+ containers:
+ - name: engine
+ env:
+ - name: switchCloud
+ value: aliyun
+ - name: cluster_id
+ value: "$(KB_CLUSTER_NAME)"
+ - name: ins_id
+ valueFrom:
+ fieldRef:
+ apiVersion: v1
+ fieldPath: metadata.name
+ - name: daemonPort
+ value: "3300"
+ - name: common_ports
+ value: '{"cdc1_port":"3009","cdc3_port":"3011","cdc2_port":"3010","cdc6_port":"3014","cdc5_port":"3013","cdc4_port":"3012"}'
+ - name: metaDb_url
+ value: "jdbc:mysql://$(GMS_SVC_NAME):$(GMS_SVC_PORT)/polardbx_meta_db?useSSL=false"
+ - name: polarx_url
+ value: "jdbc:mysql://$(CN_SVC_NAME):$(CN_SVC_PORT)/__cdc__?useSSL=false"
+ - name: metaDb_username
+ valueFrom:
+ secretKeyRef:
+ name: $(CONN_CREDENTIAL_SECRET_NAME)
+ key: username
+ optional: false
+ - name: metaDb_password
+ valueFrom:
+ secretKeyRef:
+ name: $(CONN_CREDENTIAL_SECRET_NAME)
+ key: password
+ optional: false
+ - name: polarx_username
+ valueFrom:
+ secretKeyRef:
+ name: $(CONN_CREDENTIAL_SECRET_NAME)
+ key: username
+ optional: false
+ - name: polarx_password
+ valueFrom:
+ secretKeyRef:
+ name: $(CONN_CREDENTIAL_SECRET_NAME)
+ key: password
+ optional: false
+ - name: metaDbNonEncPasswd
+ valueFrom:
+ secretKeyRef:
+ name: $(CONN_CREDENTIAL_SECRET_NAME)
+ key: password
+ optional: false
+ - name: dnPasswordKey
+ value: "$(metaDbNonEncPasswd)$(metaDbNonEncPasswd)"
+ - name: cpu_cores
+ valueFrom:
+ resourceFieldRef:
+ containerName: engine
+ resource: limits.cpu
+ - name: mem_size
+ valueFrom:
+ resourceFieldRef:
+ containerName: engine
+ resource: limits.memory
+ divisor: "1M"
+ - name: disk_size
+ value: "10240"
+ - name: disk_quota
+ value: "10240"
+ volumeMounts:
+ - name: binlog
+ mountPath: /home/admin/binlog
+ - name: log
+ mountPath: /home/admin/logs
+ - name: exporter
+ args:
+ - -web.listen-addr=:9104
+ - -web.metrics-path=/metrics
+ - -target.port=3007
+ - -target.type=CDC
+ env:
+ - name: GOMAXPROCS
+ value: "1"
+ ports:
+ - containerPort: 9104
+ name: metrics
+ protocol: TCP
+
diff --git a/deploy/polardbx/templates/clusterVersion.yaml b/deploy/polardbx/templates/clusterVersion.yaml
new file mode 100644
index 00000000000..c8271c2f64f
--- /dev/null
+++ b/deploy/polardbx/templates/clusterVersion.yaml
@@ -0,0 +1,64 @@
+apiVersion: apps.kubeblocks.io/v1alpha1
+kind: ClusterVersion
+metadata:
+ name: polardbx-{{ default .Chart.AppVersion .Values.clusterVersionOverride }}
+ labels:
+ {{- include "polardbx.labels" . | nindent 4 }}
+spec:
+ clusterDefinitionRef: polardbx
+ componentVersions:
+ - componentDefRef: gms
+ versionsContext:
+ containers:
+ - name: engine
+ image: {{ .Values.images.polardbx.repository }}/{{ .Values.images.polardbx.dn.name}}:{{.Values.images.polardbx.dn.tag}}
+ imagePullPolicy: {{ default "IfNotPresent" .Values.images.polardbx.pullPolicy }}
+ - name: exporter
+ image: {{ .Values.images.prom.repository }}/{{ .Values.images.prom.mysqld_exporter.name}}:{{.Values.images.prom.mysqld_exporter.tag}}
+ imagePullPolicy: {{ default "IfNotPresent" .Values.images.prom.pullPolicy }}
+ initContainers:
+ - name: tools-updater
+ image: {{ .Values.images.polardbx.repository }}/{{ .Values.images.polardbx.toolsUpdater.name }}:{{.Values.images.polardbx.toolsUpdater.tag}}
+ imagePullPolicy: {{ default "IfNotPresent" .Values.images.polardbx.pullPolicy }}
+ - componentDefRef: dn
+ versionsContext:
+ containers:
+ - name: engine
+ image: {{ .Values.images.polardbx.repository }}/{{ .Values.images.polardbx.dn.name}}:{{.Values.images.polardbx.dn.tag}}
+ imagePullPolicy: {{ default "IfNotPresent" .Values.images.polardbx.pullPolicy }}
+ - name: exporter
+ image: {{ .Values.images.prom.repository }}/{{ .Values.images.prom.mysqld_exporter.name}}:{{.Values.images.prom.mysqld_exporter.tag}}
+ imagePullPolicy: {{ default "IfNotPresent" .Values.images.prom.pullPolicy }}
+ initContainers:
+ - name: tools-updater
+ image: {{ .Values.images.polardbx.repository }}/{{ .Values.images.polardbx.toolsUpdater.name }}:{{.Values.images.polardbx.toolsUpdater.tag}}
+ imagePullPolicy: {{ default "IfNotPresent" .Values.images.polardbx.pullPolicy }}
+ - componentDefRef: cn
+ versionsContext:
+ containers:
+ - name: engine
+ image: {{ .Values.images.polardbx.repository }}/{{ .Values.images.polardbx.cn.name}}:{{.Values.images.polardbx.cn.tag}}
+ imagePullPolicy: {{ default "IfNotPresent" .Values.images.polardbx.pullPolicy }}
+ - name: exporter
+ image: {{ .Values.images.polardbx.repository }}/{{ .Values.images.polardbx.exporter.name}}:{{.Values.images.polardbx.exporter.tag}}
+ imagePullPolicy: {{ default "IfNotPresent" .Values.images.polardbx.pullPolicy }}
+ initContainers:
+ - name: init
+ image: {{ .Values.images.polardbx.repository }}/{{ .Values.images.polardbx.init.name }}:{{.Values.images.polardbx.init.tag}}
+ imagePullPolicy: {{ default "IfNotPresent" .Values.images.polardbx.pullPolicy }}
+ - name: metadb-init
+ image: {{ .Values.images.mysql.repository }}:{{ .Values.images.mysql.tag }}
+ imagePullPolicy: {{ default "IfNotPresent" .Values.images.mysql.pullPolicy }}
+ - componentDefRef: cdc
+ versionsContext:
+ containers:
+ - name: engine
+ image: {{ .Values.images.polardbx.repository }}/{{ .Values.images.polardbx.cdc.name}}:{{.Values.images.polardbx.cdc.tag}}
+ imagePullPolicy: {{ default "IfNotPresent" .Values.images.polardbx.pullPolicy }}
+ - name: exporter
+ image: {{ .Values.images.polardbx.repository }}/{{ .Values.images.polardbx.exporter.name}}:{{.Values.images.polardbx.exporter.tag}}
+ imagePullPolicy: {{ default "IfNotPresent" .Values.images.polardbx.pullPolicy }}
+ initContainers:
+ - name: wait-cn-ready
+ image: {{ .Values.images.mysql.repository }}:{{ .Values.images.mysql.tag }}
+ imagePullPolicy: {{ default "IfNotPresent" .Values.images.mysql.pullPolicy }}
\ No newline at end of file
diff --git a/deploy/polardbx/templates/configmap-dashboards.yaml b/deploy/polardbx/templates/configmap-dashboards.yaml
new file mode 100644
index 00000000000..686708eb6b1
--- /dev/null
+++ b/deploy/polardbx/templates/configmap-dashboards.yaml
@@ -0,0 +1,19 @@
+{{- $files := .Files.Glob "dashboards/*.json" }}
+{{- if $files }}
+apiVersion: v1
+kind: ConfigMapList
+items:
+{{- range $path, $fileContents := $files }}
+{{- $dashboardName := regexReplaceAll "(^.*/)(.*)\\.json$" $path "${2}" }}
+- apiVersion: v1
+ kind: ConfigMap
+ metadata:
+ name: {{ printf "grafana-%s" $dashboardName | trunc 63 | trimSuffix "-" }}
+ labels:
+ grafana_dashboard: "1"
+ app: {{ template "polardbx.name" $ }}-grafana
+{{ include "polardbx.labels" $ | indent 6 }}
+ data:
+ {{ $dashboardName }}.json: {{ $.Files.Get $path | toJson }}
+{{- end }}
+{{- end }}
diff --git a/deploy/polardbx/templates/scriptstemplate.yaml b/deploy/polardbx/templates/scriptstemplate.yaml
new file mode 100644
index 00000000000..7aed5b4d971
--- /dev/null
+++ b/deploy/polardbx/templates/scriptstemplate.yaml
@@ -0,0 +1,17 @@
+apiVersion: v1
+kind: ConfigMap
+metadata:
+ name: polardbx-scripts
+ labels:
+ {{- include "polardbx.labels" . | nindent 4 }}
+data:
+ xstore-post-start.sh: |-
+ {{- .Files.Get "scripts/xstore-post-start.tpl" | nindent 4 }}
+ xstore-setup.sh: |-
+ {{- .Files.Get "scripts/xstore-setup.tpl" | nindent 4 }}
+ gms-init.sql: |-
+ {{- .Files.Get "scripts/gms-init.sql" | nindent 4 }}
+ gms-metadata.tpl: |-
+ {{- .Files.Get "scripts/gms-metadata.tpl" | nindent 4 }}
+ metadb-setup.sh: |-
+ {{- .Files.Get "scripts/metadb-setup.tpl" | nindent 4 }}
\ No newline at end of file
diff --git a/deploy/polardbx/values.yaml b/deploy/polardbx/values.yaml
new file mode 100644
index 00000000000..d16bade0141
--- /dev/null
+++ b/deploy/polardbx/values.yaml
@@ -0,0 +1,61 @@
+# Default values for PolarDB-X.
+# This is a YAML-formatted file.
+# Declare variables to be passed into your templates.
+
+clusterVersionOverride: ""
+
+roleProbe:
+ failureThreshold: 2
+ periodSeconds: 1
+ timeoutSeconds: 1
+
+# Related image configurations.
+images:
+ polardbx:
+ pullPolicy: IfNotPresent
+ # Repo of polardbx default images. Default is polardbx.
+ repository: polardbx
+
+ # Images for xstore(DN) tools updater.
+ toolsUpdater:
+ name: xstore-tools
+ tag: latest
+
+ # Image for DN engine
+ dn:
+ name: polardbx-engine-2.0
+ tag: latest
+
+ # Image for CN engine
+ cn:
+ name: polardbx-sql
+ tag: latest
+
+ # Image for CN initialization
+ init:
+ name: polardbx-init
+ tag: latest
+
+ # Image for CN engine
+ cdc:
+ name: polardbx-cdc
+ tag: latest
+
+ # Image for CN&CDC exporter
+ exporter:
+ name: polardbx-exporter
+ tag: latest
+
+ # Tool image settings for gms initialization
+ mysql:
+ repository: mysql
+ pullPolicy: IfNotPresent
+ tag: "8.0.30"
+
+ # Images for DN exporter
+ prom:
+ repository: prom
+ pullPolicy: IfNotPresent
+ mysqld_exporter:
+ name: mysqld-exporter
+ tag: v0.14.0
diff --git a/deploy/postgresql/dataprotection/backup-info-collector.sh b/deploy/postgresql/dataprotection/backup-info-collector.sh
index 8844dd80ec1..2324b0b2a1d 100644
--- a/deploy/postgresql/dataprotection/backup-info-collector.sh
+++ b/deploy/postgresql/dataprotection/backup-info-collector.sh
@@ -1,5 +1,5 @@
function get_current_time() {
- curr_time=$(psql -U ${DB_USER} -h ${DB_HOST} -d postgres -t -c "SELECT now() AT TIME ZONE 'UTC'")
+ curr_time=$(psql -U ${DP_DB_USER} -h ${DP_DB_HOST} -d postgres -t -c "SELECT now() AT TIME ZONE 'UTC'")
echo $curr_time
}
@@ -11,6 +11,6 @@ function stat_and_save_backup_info() {
fi
START_TIME=$(date -d "${START_TIME}" -u '+%Y-%m-%dT%H:%M:%SZ')
STOP_TIME=$(date -d "${STOP_TIME}" -u '+%Y-%m-%dT%H:%M:%SZ')
- TOTAL_SIZE=$(du -shx ${BACKUP_DIR}|awk '{print $1}')
- echo "{\"totalSize\":\"$TOTAL_SIZE\",\"manifests\":{\"backupLog\":{\"startTime\":\"${START_TIME}\",\"stopTime\":\"${STOP_TIME}\"},\"backupTool\":{\"uploadTotalSize\":\"${TOTAL_SIZE}\"}}}" > ${BACKUP_DIR}/backup.info
+ TOTAL_SIZE=$(du -shx ${DP_BACKUP_DIR}|awk '{print $1}')
+ echo "{\"totalSize\":\"$TOTAL_SIZE\",\"timeRange\":{\"start\":\"${START_TIME}\",\"end\":\"${STOP_TIME}\"}}" > ${DP_BACKUP_DIR}/backup.info
}
\ No newline at end of file
diff --git a/deploy/postgresql/dataprotection/fetch-wal-log.sh b/deploy/postgresql/dataprotection/fetch-wal-log.sh
deleted file mode 100644
index a89b0050c76..00000000000
--- a/deploy/postgresql/dataprotection/fetch-wal-log.sh
+++ /dev/null
@@ -1,57 +0,0 @@
-function fetch-wal-log(){
- backup_log_dir=$1
- wal_destination_dir=$2
- start_wal_name=$3
- restore_time=`date -d "$4" +%s`
- pitr=$5
- echo "PITR: $pitr"
-
- if [[ ! -d ${backup_log_dir} ]]; then
- echo "ERROR: ${backup_log_dir} not exists"
- exit 1
- fi
-
- exit_fetch_wal=0 && mkdir -p $wal_destination_dir
- for dir_name in $(ls ${backup_log_dir} -l | grep ^d | awk '{print $9}' | sort); do
- if [[ $exit_fetch_wal -eq 1 ]]; then
- exit 0
- fi
-
- cd ${backup_log_dir}/${dir_name}
- # check if the latest_wal_log after the start_wal_log
- latest_wal=$(ls | sort | tail -n 1)
- if [[ $latest_wal < $start_wal_name ]]; then
- continue
- fi
-
- echo "INFO: start to fetch wal logs from ${backup_log_dir}/${dir_name}"
- for file in $(ls | sort | grep ".gz"); do
- if [[ $file < $start_wal_name ]]; then
- continue
- fi
- if [[ $pitr != "true" && $file =~ ".history" ]]; then
- # if not restored for pitr, only fetch the current timeline log
- echo "INFO: exit for new timeline."
- exit_fetch_wal=1
- break
- fi
-
- if [ ! -f $file ]; then
- echo "ERROR: $file was deleted during fetching the wal log. Please try again!"
- exit 1
- fi
- wal_name=${file%.*}
- echo "INFO: copying $wal_name"
- gunzip -c $file > ${wal_destination_dir}/$wal_name
-
- # check if the wal_log contains the restore_time logs. if ture, stop fetching
- latest_commit_time=$(pg_waldump ${wal_destination_dir}/$wal_name --rmgr=Transaction 2>/dev/null |tail -n 1|awk -F ' COMMIT ' '{print $2}'|awk -F ';' '{print $1}')
- timestamp=`date -d "$latest_commit_time" +%s`
- if [[ $latest_commit_time != "" && $timestamp > $restore_time ]]; then
- echo "INFO: exit when reaching the target time log."
- exit_fetch_wal=1
- break
- fi
- done
- done
-}
\ No newline at end of file
diff --git a/deploy/postgresql/dataprotection/pg-basebackup-backup.sh b/deploy/postgresql/dataprotection/pg-basebackup-backup.sh
new file mode 100644
index 00000000000..0fb6296841a
--- /dev/null
+++ b/deploy/postgresql/dataprotection/pg-basebackup-backup.sh
@@ -0,0 +1,12 @@
+set -e;
+if [ -d ${DP_BACKUP_DIR} ]; then
+ rm -rf ${DP_BACKUP_DIR}
+fi
+mkdir -p ${DP_BACKUP_DIR};
+export PGPASSWORD=${DP_DB_PASSWORD}
+
+START_TIME=`get_current_time`
+echo ${DP_DB_PASSWORD} | pg_basebackup -Ft -Pv -c fast -Xs -Z${COMPRESS_LEVEL} -D ${DP_BACKUP_DIR} -h ${DP_DB_HOST} -U ${DP_DB_USER} -W;
+
+# stat and save the backup information
+stat_and_save_backup_info $START_TIME
\ No newline at end of file
diff --git a/deploy/postgresql/dataprotection/pg-basebackup-restore.sh b/deploy/postgresql/dataprotection/pg-basebackup-restore.sh
new file mode 100644
index 00000000000..1be2f6ec70a
--- /dev/null
+++ b/deploy/postgresql/dataprotection/pg-basebackup-restore.sh
@@ -0,0 +1,15 @@
+set -e;
+cd ${DP_BACKUP_DIR};
+mkdir -p ${DATA_DIR};
+# compatible with gzip compression
+if [ -f base.tar.gz ];then
+ tar -xvf base.tar.gz -C ${DATA_DIR}/;
+else
+ tar -xvf base.tar -C ${DATA_DIR}/;
+fi
+if [ -f pg_wal.tar.gz ];then
+ tar -xvf pg_wal.tar.gz -C ${DATA_DIR}/pg_wal/;
+else
+ tar -xvf pg_wal.tar -C ${DATA_DIR}/pg_wal/;
+fi
+echo "done!";
\ No newline at end of file
diff --git a/deploy/postgresql/dataprotection/pitr-backup.sh b/deploy/postgresql/dataprotection/pitr-backup.sh
deleted file mode 100644
index c882af2cf55..00000000000
--- a/deploy/postgresql/dataprotection/pitr-backup.sh
+++ /dev/null
@@ -1,126 +0,0 @@
-#!/bin/bash
-export PGPASSWORD=${DB_PASSWORD}
-PSQL="psql -h ${DB_HOST} -U ${DB_USER} -d postgres"
-last_switch_wal_time=$(date +%s)
-last_purge_time=$(date +%s)
-STOP_TIME=
-switch_wal_interval=300
-
-if [[ ${SWITCH_WAL_INTERVAL_SECONDS} =~ ^[0-9]+$ ]];then
- switch_wal_interval=${SWITCH_WAL_INTERVAL_SECONDS}
-fi
-
-backup_in_secondary=
-if [ "${DP_POD_ROLE}" == "primary" ]; then
- backup_in_secondary=f
-elif [ "${DP_POD_ROLE}" == "secondary" ]; then
- backup_in_secondary=t
-fi
-
-function log() {
- msg=$1
- local curr_date=$(date -u '+%Y-%m-%d %H:%M:%S')
- echo "${curr_date} INFO: $msg"
-}
-
-function purge_expired_files() {
- # clean up expired logfiles, interval is 60s
- local curr_time=$(date +%s)
- local diff_time=$((${curr_time}-${last_purge_time}))
- if [[ -z ${LOGFILE_TTL_SECOND} || ${diff_time} -lt 60 ]]; then
- return
- fi
- retention_day=$((${LOGFILE_TTL_SECOND}/86400))
- EXPIRED_INCR_LOG=${BACKUP_DIR}/$(date -d"${retention_day} day ago" +%Y%m%d);
- if [ -d ${EXPIRED_INCR_LOG} ]; then
- rm -rf ${EXPIRED_INCR_LOG};
- fi
- last_purge_time=${curr_time}
-}
-
-function switch_wal_log() {
- local curr_time=$(date +%s)
- local diff_time=$((${curr_time}-${last_switch_wal_time}))
- if [[ ${diff_time} -lt ${switch_wal_interval} ]]; then
- return
- fi
- LAST_TRANS=$(pg_waldump $(${PSQL} -Atc "select pg_walfile_name(pg_current_wal_lsn())") --rmgr=Transaction 2>/dev/null |tail -n 1)
- if [ "${LAST_TRANS}" != "" ] && [ "$(find ${LOG_DIR}/archive_status/ -name '*.ready')" = "" ]; then
- log "start to switch wal file"
- ${PSQL} -c "select pg_switch_wal()"
- for i in $(seq 1 60); do
- if [ "$(find ${LOG_DIR}/archive_status/ -name '*.ready')" != "" ]; then
- log "switch wal file successfully"
- break;
- fi
- sleep 1
- done
- fi
- last_switch_wal_time=${curr_time}
-}
-
-function upload_wal_log() {
- TODAY_INCR_LOG=${BACKUP_DIR}/$(date +%Y%m%d);
- mkdir -p ${TODAY_INCR_LOG};
- cd ${LOG_DIR}
- for i in $(ls -tr ./archive_status/ | grep .ready); do
- wal_name=${i%.*}
- LOG_STOP_TIME=$(pg_waldump ${wal_name} --rmgr=Transaction 2>/dev/null |tail -n 1|awk -F ' COMMIT ' '{print $2}'|awk -F ';' '{print $1}')
- if [[ ! -z $LOG_STOP_TIME ]];then
- STOP_TIME=$(date -d "${LOG_STOP_TIME}" -u '+%Y-%m-%dT%H:%M:%SZ')
- fi
- if [ -f ${wal_name} ]; then
- log "upload ${wal_name}"
- gzip -kqc ${wal_name} > ${TODAY_INCR_LOG}/${wal_name}.gz;
- mv -f ./archive_status/${i} ./archive_status/${wal_name}.done;
- fi
- done
-}
-
-function save_backup_status() {
- TOTAL_SIZE=$(du -shx ${BACKUP_DIR}|awk '{print $1}')
- if [[ -z ${STOP_TIME} ]];then
- echo "{\"totalSize\":\"${TOTAL_SIZE}\",\"manifests\":{\"backupTool\":{\"uploadTotalSize\":\"${TOTAL_SIZE}\"}}}" > ${BACKUP_DIR}/backup.info
- else
- echo "{\"totalSize\":\"${TOTAL_SIZE}\",\"manifests\":{\"backupLog\":{\"stopTime\":\"${STOP_TIME}\"},\"backupTool\":{\"uploadTotalSize\":\"${TOTAL_SIZE}\"}}}" > ${BACKUP_DIR}/backup.info
- fi
-}
-
-function check_pg_process() {
- is_ok=false
- for ((i=1;i<4;i++));do
- is_secondary=$(${PSQL} -Atc "select pg_is_in_recovery()")
- if [[ $? -eq 0 && (-z ${backup_in_secondary} || "${backup_in_secondary}" == "${is_secondary}") ]]; then
- is_ok=true
- break
- fi
- echo "Warning: target backup pod/${DP_TARGET_POD_NAME} is not OK, target role: ${DP_POD_ROLE}, pg_is_in_recovery: ${is_secondary}, retry detection!"
- sleep 1
- done
- if [[ ${is_ok} == "false" ]];then
- echo "ERROR: target backup pod/${DP_TARGET_POD_NAME} is not OK, target role: ${DP_POD_ROLE}, pg_is_in_recovery: ${is_secondary}!"
- exit 1
- fi
-}
-
-# trap term signal
-trap "echo 'Terminating...' && sync && exit 0" TERM
-log "start to archive wal logs"
-while true; do
-
- # check if pg process is ok
- check_pg_process
-
- # switch wal log
- switch_wal_log
-
- # upload wal log
- upload_wal_log
-
- # save backup status which will be updated to `backup` CR by the sidecar
- save_backup_status
-
- # purge the expired wal logs
- purge_expired_files
- sleep ${DP_INTERVAL_SECONDS}
-done
\ No newline at end of file
diff --git a/deploy/postgresql/dataprotection/pitr-restore.sh b/deploy/postgresql/dataprotection/pitr-restore.sh
deleted file mode 100644
index 0bf4ed28a2a..00000000000
--- a/deploy/postgresql/dataprotection/pitr-restore.sh
+++ /dev/null
@@ -1,30 +0,0 @@
-
-if [ -d ${DATA_DIR}.old ];
- then echo "${DATA_DIR}.old directory already exists, skip restore.";
- exit 0;
-fi
-
-mkdir -p ${PITR_DIR};
-
-latest_wal=$(ls ${DATA_DIR}/pg_wal -lI "*.history" | grep ^- | awk '{print $9}' | sort | tail -n 1)
-start_wal_log=`basename $latest_wal`
-
-echo "fetch-wal-log ${BACKUP_DIR} ${PITR_DIR} ${start_wal_log} \"${KB_RECOVERY_TIME}\" true"
-fetch-wal-log ${BACKUP_DIR} ${PITR_DIR} ${start_wal_log} "${KB_RECOVERY_TIME}" true
-
-chmod 777 -R ${PITR_DIR};
-touch ${DATA_DIR}/recovery.signal;
-mkdir -p ${CONF_DIR};
-chmod 777 -R ${CONF_DIR};
-mkdir -p ${RESTORE_SCRIPT_DIR};
-echo "#!/bin/bash" > ${RESTORE_SCRIPT_DIR}/kb_restore.sh;
-echo "[[ -d '${DATA_DIR}.old' ]] && mv -f ${DATA_DIR}.old/* ${DATA_DIR}/;" >> ${RESTORE_SCRIPT_DIR}/kb_restore.sh;
-echo "sync;" >> ${RESTORE_SCRIPT_DIR}/kb_restore.sh;
-chmod +x ${RESTORE_SCRIPT_DIR}/kb_restore.sh;
-echo "restore_command='case "%f" in *history) cp ${PITR_DIR}/%f %p ;; *) mv ${PITR_DIR}/%f %p ;; esac'" > ${CONF_DIR}/recovery.conf;
-echo "recovery_target_time='${KB_RECOVERY_TIME}'" >> ${CONF_DIR}/recovery.conf;
-echo "recovery_target_action='promote'" >> ${CONF_DIR}/recovery.conf;
-echo "recovery_target_timeline='latest'" >> ${CONF_DIR}/recovery.conf;
-mv ${DATA_DIR} ${DATA_DIR}.old;
-echo "done.";
-sync;
\ No newline at end of file
diff --git a/deploy/postgresql/templates/actionset-pgbasebackup.yaml b/deploy/postgresql/templates/actionset-pgbasebackup.yaml
new file mode 100644
index 00000000000..94e8904d0da
--- /dev/null
+++ b/deploy/postgresql/templates/actionset-pgbasebackup.yaml
@@ -0,0 +1,38 @@
+apiVersion: dataprotection.kubeblocks.io/v1alpha1
+kind: ActionSet
+metadata:
+ name: postgres-basebackup
+ labels:
+ clusterdefinition.kubeblocks.io/name: postgresql
+ {{- include "postgresql.labels" . | nindent 4 }}
+spec:
+ backupType: Full
+ env:
+ - name: DATA_DIR
+ value: {{ .Values.dataMountPath }}/pgroot/data
+ - name: COMPRESS_LEVEL
+ value: "0"
+ backup:
+ preBackup: []
+ postBackup: []
+ backupData:
+ image: {{ .Values.image.registry | default "docker.io" }}/{{ .Values.image.repository }}:{{ .Values.image.tag }}
+ runOnTargetPodNode: false
+ command:
+ - bash
+ - -c
+ - |
+ {{- .Files.Get "dataprotection/backup-info-collector.sh" | nindent 8 }}
+ {{- .Files.Get "dataprotection/pg-basebackup-backup.sh" | nindent 8 }}
+ syncProgress:
+ enabled: true
+ intervalSeconds: 5
+ restore:
+ prepareData:
+ image: {{ .Values.image.registry | default "docker.io" }}/{{ .Values.image.repository }}:{{ .Values.image.tag }}
+ command:
+ - bash
+ - -c
+ - |
+ {{- .Files.Get "dataprotection/pg-basebackup-restore.sh" | nindent 8 }}
+ postReady: []
diff --git a/deploy/postgresql/templates/backuppolicytemplate.yaml b/deploy/postgresql/templates/backuppolicytemplate.yaml
index ed9c871ce7b..dc0467666bf 100644
--- a/deploy/postgresql/templates/backuppolicytemplate.yaml
+++ b/deploy/postgresql/templates/backuppolicytemplate.yaml
@@ -5,53 +5,30 @@ metadata:
labels:
clusterdefinition.kubeblocks.io/name: postgresql
{{- include "postgresql.labels" . | nindent 4 }}
- annotations:
- dataprotection.kubeblocks.io/reconfigure-ref: |
- {
- "name": "postgresql-configuration",
- "key": "postgresql.conf",
- "enable": {
- "logfile": [{"key": "archive_command","value": "''"}]
- },
- "disable": {
- "logfile": [{ "key": "archive_command","value": "'/bin/true'"}]
- }
- }
spec:
clusterDefinitionRef: postgresql
backupPolicies:
- componentDefRef: postgresql
- retention:
- ttl: 7d
- schedule:
- startingDeadlineMinutes: 120
- snapshot:
- enable: false
- cronExpression: "0 18 * * *"
- datafile:
- enable: false
- cronExpression: "0 18 * * *"
- logfile:
- enable: false
- cronExpression: "*/2 * * * *"
- snapshot:
- target:
- connectionCredentialKey:
- passwordKey: password
- usernameKey: username
- hooks:
- containerName: postgresql
- preCommands:
- - psql -c "CHECKPOINT;"
- datafile:
- backupToolName: postgres-basebackup
- backupStatusUpdates:
- - updateStage: post
- useTargetPodServiceAccount: true
- logfile:
- backupToolName: postgres-pitr
- target:
- role: primary
- backupStatusUpdates:
- - updateStage: post
- useTargetPodServiceAccount: true
\ No newline at end of file
+ retentionPeriod: 7d
+ target:
+ role: secondary
+ backupMethods:
+ - name: pg-basebackup
+ snapshotVolumes: false
+ actionSetName: postgres-basebackup
+ targetVolumes:
+ volumeMounts:
+ - name: data
+ mountPath: {{ .Values.dataMountPath }}
+ - name: volume-snapshot
+ snapshotVolumes: true
+ targetVolumes:
+ volumes:
+ - data
+ schedules:
+ - backupMethod: pg-basebackup
+ enabled: false
+ cronExpression: "0 18 * * *"
+ - backupMethod: volume-snapshot
+ enabled: false
+ cronExpression: "0 18 * * *"
\ No newline at end of file
diff --git a/deploy/postgresql/templates/backuptool-pgbasebackup.yaml b/deploy/postgresql/templates/backuptool-pgbasebackup.yaml
deleted file mode 100644
index 1b46757693b..00000000000
--- a/deploy/postgresql/templates/backuptool-pgbasebackup.yaml
+++ /dev/null
@@ -1,56 +0,0 @@
-apiVersion: dataprotection.kubeblocks.io/v1alpha1
-kind: BackupTool
-metadata:
- name: postgres-basebackup
- labels:
- clusterdefinition.kubeblocks.io/name: postgresql
- {{- include "postgresql.labels" . | nindent 4 }}
-spec:
- image: {{ .Values.image.registry | default "docker.io" }}/{{ .Values.image.repository }}:{{ .Values.image.tag }}
- deployKind: job
- env:
- - name: DATA_DIR
- value: /home/postgres/pgdata/pgroot/data
- physical:
- restoreCommands:
- - sh
- - -c
- - |
- #!/bin/sh
- set -e;
- cd ${BACKUP_DIR};
- mkdir -p ${DATA_DIR};
- # compatible with gzip compression for version 0.5.0
- if [ -f base.tar.gz ];then
- tar -xvf base.tar.gz -C ${DATA_DIR}/;
- else
- tar -xvf base.tar -C ${DATA_DIR}/;
- fi
- if [ -f pg_wal.tar.gz ];then
- tar -xvf pg_wal.tar.gz -C ${DATA_DIR}/pg_wal/;
- else
- tar -xvf pg_wal.tar -C ${DATA_DIR}/pg_wal/;
- fi
- echo "done!";
- incrementalRestoreCommands: []
- logical:
- restoreCommands: []
- incrementalRestoreCommands: []
- backupCommands:
- - bash
- - -c
- - |
- set -e;
- if [ -d ${BACKUP_DIR} ]; then
- rm -rf ${BACKUP_DIR}
- fi
- mkdir -p ${BACKUP_DIR};
- export PGPASSWORD=${DB_PASSWORD}
- {{- .Files.Get "dataprotection/backup-info-collector.sh" | nindent 6 }}
-
- START_TIME=`get_current_time`
- echo ${DB_PASSWORD} | pg_basebackup -Ft -Pv -c fast -Xs -D ${BACKUP_DIR} -h ${DB_HOST} -U ${DB_USER} -W;
-
- # stat and save the backup information
- stat_and_save_backup_info $START_TIME
- incrementalBackupCommands: []
diff --git a/deploy/postgresql/templates/backuptool-pitr.yaml b/deploy/postgresql/templates/backuptool-pitr.yaml
deleted file mode 100644
index 05474e68360..00000000000
--- a/deploy/postgresql/templates/backuptool-pitr.yaml
+++ /dev/null
@@ -1,56 +0,0 @@
-apiVersion: dataprotection.kubeblocks.io/v1alpha1
-kind: BackupTool
-metadata:
- labels:
- clusterdefinition.kubeblocks.io/name: postgresql
- kubeblocks.io/backup-tool-type: pitr
- {{- include "postgresql.labels" . | nindent 4 }}
- name: postgres-pitr
-spec:
- deployKind: statefulSet
- env:
- - name: VOLUME_DATA_DIR
- value: /home/postgres/pgdata
- - name: RESTORE_SCRIPT_DIR
- value: "$(VOLUME_DATA_DIR)/kb_restore"
- - name: PITR_DIR
- value: "$(VOLUME_DATA_DIR)/pitr"
- - name: DATA_DIR
- value: "$(VOLUME_DATA_DIR)/pgroot/data"
- - name: CONF_DIR
- value: "$(VOLUME_DATA_DIR)/conf"
- - name: TIME_FORMAT
- value: 2006-01-02 15:04:05 MST
- - name: LOG_DIR
- value: $(VOLUME_DATA_DIR)/pgroot/data/pg_wal
- - name: DP_POD_ROLE
- # TODO input by backup policy
- value: primary
- - name: DP_INTERVAL_SECONDS
- value: "10"
- - name: SWITCH_WAL_INTERVAL_SECONDS
- value: "600"
- image: ""
- logical:
- restoreCommands:
- - sh
- - -c
- - |
- set -e;
- rm -f ${CONF_DIR}/recovery.conf;
- rm -rf ${PITR_DIR};
- physical:
- restoreCommands:
- - bash
- - -c
- - |
- #!/bin/bash
- set -e;
- {{- .Files.Get "dataprotection/fetch-wal-log.sh" | nindent 8 }}
- {{- .Files.Get "dataprotection/pitr-restore.sh" | nindent 8 }}
- backupCommands:
- - bash
- - -c
- - |
- {{- .Files.Get "dataprotection/pitr-backup.sh" | nindent 6 }}
- type: pitr
\ No newline at end of file
diff --git a/deploy/postgresql/templates/backuptool-wal-g.yaml b/deploy/postgresql/templates/backuptool-wal-g.yaml
deleted file mode 100644
index 1466406bd16..00000000000
--- a/deploy/postgresql/templates/backuptool-wal-g.yaml
+++ /dev/null
@@ -1,76 +0,0 @@
-apiVersion: dataprotection.kubeblocks.io/v1alpha1
-kind: BackupTool
-metadata:
- name: postgres-wal-g
- labels:
- clusterdefinition.kubeblocks.io/name: postgresql
-spec:
- image: {{ .Values.image.registry | default "docker.io" }}/{{ .Values.image.repository }}:{{ .Values.image.tag }}
- deployKind: job
- env:
- - name: DATA_DIR
- value: /home/postgres/pgdata/pgroot/data
- - name: WAL_DIR
- value: $(DATA_DIR)/pg_wal
- - name: WALG_PG_WAL_SIZE
- value: "16"
- - name: WALG_TAR_SIZE_THRESHOLD
- value: "4294967296"
- - name: WALG_UPLOAD_DISK_CONCURRENCY
- value: "8"
- physical:
- relyOnLogfile: true
- restoreCommands:
- - bash
- - -c
- - |
- #!/bin/bash
- # NOTE: this basebackup is only supported when pitr is enabled and rely on archive logs.
- # if archive logs are deleted, it will cause the recovery failed from this backup.
- set -e;
-
- {{- .Files.Get "dataprotection/fetch-wal-log.sh" | nindent 8 }}
-
- # fetch base backup
- mkdir -p ${DATA_DIR};
- WALG_FILE_PREFIX=${BACKUP_DIR} wal-g backup-fetch ${DATA_DIR} LATEST
-
- if [[ ! -z ${BACKUP_LOGFILE_DIR} ]]; then
- # get start wal log
- start_wal_location=$(cat ${DATA_DIR}/backup_label | grep "START WAL LOCATION")
- start_wal_log=${start_wal_location#*file } && start_wal_log=${start_wal_log/)/}
-
- # fetch wal logs from archive dir
- echo "fetch-wal-log ${BACKUP_LOGFILE_DIR} ${WAL_DIR} ${start_wal_log} \"${BACKUP_STOP_TIME}\" false"
- fetch-wal-log ${BACKUP_LOGFILE_DIR} ${WAL_DIR} ${start_wal_log} "${BACKUP_STOP_TIME}" false
- fi
- echo "done!";
- incrementalRestoreCommands: []
- logical:
- restoreCommands: []
- incrementalRestoreCommands: []
- backupCommands:
- - bash
- - -c
- - |-
- set -e;
- if [ -d ${BACKUP_DIR} ]; then
- rm -rf ${BACKUP_DIR}
- fi
- mkdir -p ${BACKUP_DIR}
- export PGPASSWORD=${DB_PASSWORD}
- {{- .Files.Get "dataprotection/backup-info-collector.sh" | nindent 6 }}
-
- START_TIME=`get_current_time`
- WALG_FILE_PREFIX=${BACKUP_DIR} PGHOST=${DB_HOST} PGUSER=${DB_USER} PGPASSWORD=${DB_PASSWORD} PGPORT=5432 wal-g backup-push ${DATA_DIR}
-
- STOP_TIME=""
- stop_sentinel_file=$(find ${BACKUP_DIR}/basebackups_005 -name "*backup_stop_sentinel.json")
- if [ -f $stop_sentinel_file ];then
- result_json=$(cat $stop_sentinel_file)
- STOP_TIME=$(echo $result_json | jq -r ".FinishTime")
- START_TIME=$(echo $result_json | jq -r ".StartTime")
- fi
- # stat and save the backup information
- stat_and_save_backup_info $START_TIME $STOP_TIME
- incrementalBackupCommands: []
diff --git a/deploy/postgresql/templates/clusterdefinition.yaml b/deploy/postgresql/templates/clusterdefinition.yaml
index 5dff801861a..6543c773067 100644
--- a/deploy/postgresql/templates/clusterdefinition.yaml
+++ b/deploy/postgresql/templates/clusterdefinition.yaml
@@ -113,7 +113,7 @@ spec:
- /kb-scripts/init_container.sh
volumeMounts:
- name: data
- mountPath: /home/postgres/pgdata
+ mountPath: {{ .Values.dataMountPath }}
- name: postgresql-config
mountPath: /home/postgres/conf
- name: scripts
@@ -145,7 +145,7 @@ spec:
- name: dshm
mountPath: /dev/shm
- name: data
- mountPath: /home/postgres/pgdata
+ mountPath: {{ .Values.dataMountPath }}
- name: postgresql-config
mountPath: /home/postgres/conf
- name: scripts
@@ -171,7 +171,7 @@ spec:
- name: KUBERNETES_LABELS
value: '{"app.kubernetes.io/instance":"$(KB_CLUSTER_NAME)","apps.kubeblocks.io/component-name":"$(KB_COMP_NAME)"}'
- name: RESTORE_DATA_DIR
- value: /home/postgres/pgdata/kb_restore
+ value: {{ .Values.dataMountPath }}/kb_restore
- name: KB_PG_CONFIG_PATH
value: /home/postgres/conf/postgresql.conf
- name: SPILO_CONFIGURATION
@@ -183,7 +183,7 @@ spec:
- name: ALLOW_NOSSL
value: "true"
- name: PGROOT
- value: /home/postgres/pgdata/pgroot
+ value: {{ .Values.dataMountPath }}/pgroot
- name: POD_IP
valueFrom:
fieldRef:
diff --git a/deploy/postgresql/values.yaml b/deploy/postgresql/values.yaml
index acc91791a7a..86a01ebc24b 100644
--- a/deploy/postgresql/values.yaml
+++ b/deploy/postgresql/values.yaml
@@ -111,5 +111,7 @@ pgbouncer:
tag: 1.19.0
pullPolicy: IfNotPresent
+dataMountPath: /home/postgres/pgdata
+
logConfigs:
running: /home/postgres/pgdata/pgroot/data/log/postgresql-*
\ No newline at end of file
diff --git a/deploy/qdrant/scripts/qdrant-backup.sh b/deploy/qdrant/scripts/qdrant-backup.sh
index 45d8766182b..a8e5da4be97 100644
--- a/deploy/qdrant/scripts/qdrant-backup.sh
+++ b/deploy/qdrant/scripts/qdrant-backup.sh
@@ -1,17 +1,20 @@
#!/usr/bin/env bash
set -e
-mkdir -p ${BACKUP_DIR}
-endpoint=http://${DB_HOST}:6333
+mkdir -p ${DP_BACKUP_DIR}
+endpoint=http://${DP_DB_HOST}:6333
snapshot=`curl -XPOST ${endpoint}/snapshots`
status=`echo ${snapshot} | jq '.status'`
-if [ "${status}" != "ok" ]; then
+if [ "${status}" != "ok" ] && [ "${status}" != "\"ok\"" ]; then
echo "backup failed, status: ${status}"
exit 1
fi
name=`echo ${snapshot} | jq '.result.name'`
-curl ${endpoint}/snapshots/${name} --output ${BACKUP_DIR}/${BACKUP_NAME}.snapshot
+curl ${endpoint}/snapshots/${name} --output ${DP_BACKUP_DIR}/${DP_BACKUP_NAME}.snapshot
-curl -XDELETE ${endpoint}/snapshots/${name}
\ No newline at end of file
+curl -XDELETE ${endpoint}/snapshots/${name}
+
+TOTAL_SIZE=$(du -shx ${DP_BACKUP_DIR}|awk '{print $1}')
+echo "{\"totalSize\":\"$TOTAL_SIZE\"}" > ${DP_BACKUP_DIR}/backup.info
\ No newline at end of file
diff --git a/deploy/qdrant/scripts/qdrant-restore.sh b/deploy/qdrant/scripts/qdrant-restore.sh
index bb7d6ecd909..fc0d3007cee 100644
--- a/deploy/qdrant/scripts/qdrant-restore.sh
+++ b/deploy/qdrant/scripts/qdrant-restore.sh
@@ -9,7 +9,7 @@ if [ ! -z "${res}" ]; then
fi
# start qdrant restore process
-qdrant --storage-snapshot ${BACKUP_DIR}/${BACKUP_NAME} --config-path /qdrant/config/config.yaml --force-snapshot --uri http://localhost:6333 &
+qdrant --storage-snapshot ${DP_BACKUP_DIR}/${DP_BACKUP_NAME} --config-path /qdrant/config/config.yaml --force-snapshot --uri http://localhost:6333 &
# wait until restore finished
until curl http://localhost:6333/cluster; do sleep 1; done
diff --git a/deploy/qdrant/templates/actionset-datafile.yaml b/deploy/qdrant/templates/actionset-datafile.yaml
new file mode 100644
index 00000000000..6ea586031b4
--- /dev/null
+++ b/deploy/qdrant/templates/actionset-datafile.yaml
@@ -0,0 +1,35 @@
+apiVersion: dataprotection.kubeblocks.io/v1alpha1
+kind: ActionSet
+metadata:
+ name: qdrant-snapshot
+ labels:
+ clusterdefinition.kubeblocks.io/name: qdrant
+ {{- include "qdrant.labels" . | nindent 4 }}
+spec:
+ backupType: Full
+ env:
+ - name: DATA_DIR
+ value: {{ .Values.dataMountPath }}/
+ backup:
+ preBackup: []
+ postBackup: []
+ backupData:
+ image: apecloud/curl-jq:latest
+ runOnTargetPodNode: false
+ command:
+ - sh
+ - -c
+ - |
+ {{- .Files.Get "scripts/qdrant-backup.sh" | nindent 8 }}
+ syncProgress:
+ enabled: true
+ intervalSeconds: 5
+ restore:
+ prepareData:
+ image: apecloud/curl-jq:latest
+ command:
+ - sh
+ - -c
+ - |
+ {{- .Files.Get "scripts/qdrant-restore.sh" | nindent 8 }}
+ postReady: []
\ No newline at end of file
diff --git a/deploy/qdrant/templates/backuppolicytemplate.yaml b/deploy/qdrant/templates/backuppolicytemplate.yaml
index f3501dda115..5b58864a1dc 100644
--- a/deploy/qdrant/templates/backuppolicytemplate.yaml
+++ b/deploy/qdrant/templates/backuppolicytemplate.yaml
@@ -9,16 +9,24 @@ spec:
clusterDefinitionRef: qdrant
backupPolicies:
- componentDefRef: qdrant
- retention:
- ttl: 7d
- schedule:
- snapshot:
- enable: false
- cronExpression: "0 18 * * 0"
- snapshot:
- target:
- connectionCredentialKey:
- passwordKey: password
- usernameKey: username
- datafile:
- backupToolName: qdrant-snapshot
\ No newline at end of file
+ retentionPeriod: 7d
+ backupMethods:
+ - name: datafile
+ snapshotVolumes: false
+ actionSetName: qdrant-snapshot
+ targetVolumes:
+ volumeMounts:
+ - name: data
+ mountPath: {{ .Values.dataMountPath }}
+ - name: volume-snapshot
+ snapshotVolumes: true
+ targetVolumes:
+ volumes:
+ - data
+ schedules:
+ - backupMethod: datafile
+ enabled: false
+ cronExpression: "0 18 * * 0"
+ - backupMethod: volume-snapshot
+ enabled: false
+ cronExpression: "0 18 * * 0"
\ No newline at end of file
diff --git a/deploy/qdrant/templates/backuptool.yaml b/deploy/qdrant/templates/backuptool.yaml
deleted file mode 100644
index 878065bf496..00000000000
--- a/deploy/qdrant/templates/backuptool.yaml
+++ /dev/null
@@ -1,25 +0,0 @@
-apiVersion: dataprotection.kubeblocks.io/v1alpha1
-kind: BackupTool
-metadata:
- name: qdrant-snapshot
- labels:
- clusterdefinition.kubeblocks.io/name: qdrant
- {{- include "qdrant.labels" . | nindent 4 }}
-spec:
- image: apecloud/curl-jq:latest
- deployKind: job
- env:
- - name: DATA_DIR
- value: /qdrant/storage/
- physical:
- restoreCommands:
- - |-
- {{- .Files.Get "scripts/qdrant-restore.sh" | nindent 8 }}
- incrementalRestoreCommands: []
- logical:
- restoreCommands: []
- incrementalRestoreCommands: []
- backupCommands:
- - |-
- {{- .Files.Get "scripts/qdrant-backup.sh" | nindent 6 }}
- incrementalBackupCommands: []
diff --git a/deploy/qdrant/templates/clusterdefinition.yaml b/deploy/qdrant/templates/clusterdefinition.yaml
index 418f7dcbc01..b613df43501 100644
--- a/deploy/qdrant/templates/clusterdefinition.yaml
+++ b/deploy/qdrant/templates/clusterdefinition.yaml
@@ -130,7 +130,7 @@ spec:
volumeMounts:
- mountPath: /qdrant/config/
name: qdrant-config
- - mountPath: /qdrant/storage
+ - mountPath: {{ .Values.dataMountPath }}
name: data
- mountPath: /qdrant/scripts
name: scripts
diff --git a/deploy/qdrant/values.yaml b/deploy/qdrant/values.yaml
index 118b1cd1be0..82bb5236a27 100644
--- a/deploy/qdrant/values.yaml
+++ b/deploy/qdrant/values.yaml
@@ -28,4 +28,6 @@ images:
## @param debugEnabled enables containers' debug logging
##
-debugEnabled: true
\ No newline at end of file
+debugEnabled: true
+
+dataMountPath: /qdrant/storage
\ No newline at end of file
diff --git a/deploy/redis/dataprotection/backup.sh b/deploy/redis/dataprotection/backup.sh
new file mode 100644
index 00000000000..a60057b4a7e
--- /dev/null
+++ b/deploy/redis/dataprotection/backup.sh
@@ -0,0 +1,19 @@
+set -e
+connect_url="redis-cli -h ${DP_DB_HOST} -p ${DP_DB_PORT} -a ${DP_DB_PASSWORD}"
+last_save=$(${connect_url} LASTSAVE)
+echo "INFO: start BGSAVE"
+${connect_url} BGSAVE
+echo "INFO: wait for saving rdb successfully"
+while true; do
+ end_save=$(${connect_url} LASTSAVE)
+ if [ $end_save -ne $last_save ];then
+ break
+ fi
+ sleep 1
+done
+echo "INFO: start to save data file..."
+mkdir -p ${DP_BACKUP_DIR} && cd ${DATA_DIR}
+tar -czvf ${DP_BACKUP_DIR}/${DP_BACKUP_NAME}.tar.gz ./
+echo "INFO: save data file successfully"
+TOTAL_SIZE=$(du -shx ${DP_BACKUP_DIR}|awk '{print $1}')
+echo "{\"totalSize\":\"$TOTAL_SIZE\"}" > ${DP_BACKUP_DIR}/backup.info && sync
\ No newline at end of file
diff --git a/deploy/redis/dataprotection/restore.sh b/deploy/redis/dataprotection/restore.sh
new file mode 100644
index 00000000000..c1e72b0f2ed
--- /dev/null
+++ b/deploy/redis/dataprotection/restore.sh
@@ -0,0 +1,12 @@
+set -e
+mkdir -p ${DATA_DIR}
+res=`find ${DATA_DIR} -type f`
+data_protection_file=${DATA_DIR}/.kb-data-protection
+if [ ! -z "${res}" ] && [ ! -f ${data_protection_file} ]; then
+ echo "${DATA_DIR} is not empty! Please make sure that the directory is empty before restoring the backup."
+ exit 1
+fi
+# touch placeholder file
+touch ${data_protection_file}
+tar -xvf ${DP_BACKUP_DIR}/${DP_BACKUP_NAME}.tar.gz -C ${DATA_DIR}
+rm -rf ${data_protection_file} && sync
\ No newline at end of file
diff --git a/deploy/redis/templates/backupactionset.yaml b/deploy/redis/templates/backupactionset.yaml
new file mode 100644
index 00000000000..f406fa92461
--- /dev/null
+++ b/deploy/redis/templates/backupactionset.yaml
@@ -0,0 +1,37 @@
+apiVersion: dataprotection.kubeblocks.io/v1alpha1
+kind: ActionSet
+metadata:
+ name: redis-physical-backup
+ labels:
+ clusterdefinition.kubeblocks.io/name: redis
+ {{- include "redis.labels" . | nindent 4 }}
+spec:
+ backupType: Full
+ env:
+ - name: DATA_DIR
+ value: {{ .Values.dataMountPath }}
+ - name: DP_DB_PORT
+ value: "6379"
+ backup:
+ preBackup: []
+ postBackup: []
+ backupData:
+ image: {{ include "redis.image" . }}
+ runOnTargetPodNode: true
+ syncProgress:
+ enabled: true
+ intervalSeconds: 5
+ command:
+ - bash
+ - -c
+ - |
+ {{- .Files.Get "dataprotection/backup.sh" | nindent 8 }}
+ restore:
+ prepareData:
+ image: {{ include "redis.image" . }}
+ command:
+ - sh
+ - -c
+ - |
+ {{- .Files.Get "dataprotection/restore.sh" | nindent 8 }}
+ postReady: []
diff --git a/deploy/redis/templates/backuppolicytemplate.yaml b/deploy/redis/templates/backuppolicytemplate.yaml
index 84b1fe46435..251712f4481 100644
--- a/deploy/redis/templates/backuppolicytemplate.yaml
+++ b/deploy/redis/templates/backuppolicytemplate.yaml
@@ -9,26 +9,26 @@ spec:
clusterDefinitionRef: redis
backupPolicies:
- componentDefRef: redis
- retention:
- ttl: 7d
- schedule:
- startingDeadlineMinutes: 120
- snapshot:
- enable: false
- cronExpression: "0 18 * * 0"
- datafile:
- enable: false
- cronExpression: "0 18 * * 0"
- snapshot:
- target:
- connectionCredentialKey:
- passwordKey: password
- usernameKey: username
- datafile:
- backupToolName: redis-physical-backup-tool
- backupsHistoryLimit: 7
- target:
- role: secondary
- backupStatusUpdates:
- - updateStage: post
- useTargetPodServiceAccount: true
\ No newline at end of file
+ retentionPeriod: 7d
+ target:
+ role: secondary
+ backupMethods:
+ - name: datafile
+ snapshotVolumes: false
+ actionSetName: redis-physical-backup
+ targetVolumes:
+ volumeMounts:
+ - name: data
+ mountPath: {{ .Values.dataMountPath }}
+ - name: volume-snapshot
+ snapshotVolumes: true
+ targetVolumes:
+ volumes:
+ - data
+ schedules:
+ - backupMethod: datafile
+ enabled: false
+ cronExpression: "0 18 * * 0"
+ - backupMethod: volume-snapshot
+ enabled: false
+ cronExpression: "0 18 * * 0"
\ No newline at end of file
diff --git a/deploy/redis/templates/backuptool.yaml b/deploy/redis/templates/backuptool.yaml
deleted file mode 100644
index a6a82cfe881..00000000000
--- a/deploy/redis/templates/backuptool.yaml
+++ /dev/null
@@ -1,54 +0,0 @@
-apiVersion: dataprotection.kubeblocks.io/v1alpha1
-kind: BackupTool
-metadata:
- name: redis-physical-backup-tool
- labels:
- clusterdefinition.kubeblocks.io/name: redis
- {{- include "redis.labels" . | nindent 4 }}
-spec:
- image: {{ include "redis.image" . }}
- deployKind: job
- env:
- - name: DATA_DIR
- value: /data
- physical:
- restoreCommands:
- - sh
- - -c
- - |
- set -e
- mkdir -p ${DATA_DIR}
- res=`find ${DATA_DIR} -type f`
- data_protection_file=${DATA_DIR}/.kb-data-protection
- if [ ! -z "${res}" ] && [ ! -f ${data_protection_file} ]; then
- echo "${DATA_DIR} is not empty! Please make sure that the directory is empty before restoring the backup."
- exit 1
- fi
- touch ${data_protection_file} && sync
- tar -xvf ${BACKUP_DIR}/${BACKUP_NAME}.tar.gz -C ${DATA_DIR}
- rm -rf ${data_protection_file} && sync
- logical:
- restoreCommands: []
- backupCommands:
- - bash
- - -c
- - |
- set -e
- connect_url="redis-cli -h ${DB_HOST} -p 6379 -a ${DB_PASSWORD}"
- last_save=$(${connect_url} LASTSAVE)
- echo "INFO: start BGSAVE"
- ${connect_url} BGSAVE
- echo "INFO: wait for saving rdb successfully"
- while true; do
- end_save=$(${connect_url} LASTSAVE)
- if [ $end_save -ne $last_save ];then
- break
- fi
- sleep 1
- done
- echo "INFO: start to save data file..."
- mkdir -p ${BACKUP_DIR} && cd ${DATA_DIR} && sync
- tar -czvf ${BACKUP_DIR}/${BACKUP_NAME}.tar.gz ./
- echo "INFO: save data file successfully" && sync
- TOTAL_SIZE=$(du -shx ${BACKUP_DIR}|awk '{print $1}')
- echo "{\"totalSize\":\"$TOTAL_SIZE\",\"manifests\":{\"backupTool\":{\"uploadTotalSize\":\"${TOTAL_SIZE}\"}}}" > ${BACKUP_DIR}/backup.info && sync
diff --git a/deploy/redis/templates/clusterdefinition.yaml b/deploy/redis/templates/clusterdefinition.yaml
index 766917176f8..2fe21b69ce1 100644
--- a/deploy/redis/templates/clusterdefinition.yaml
+++ b/deploy/redis/templates/clusterdefinition.yaml
@@ -64,7 +64,7 @@ spec:
containerPort: 6379
volumeMounts:
- name: data
- mountPath: /data
+ mountPath: {{ .Values.dataMountPath }}
- name: redis-config
mountPath: /etc/conf
- name: scripts
@@ -231,7 +231,7 @@ spec:
imagePullPolicy: IfNotPresent
volumeMounts:
- name: data
- mountPath: /data
+ mountPath: {{ .Values.dataMountPath }}
- name: redis-config
mountPath: /etc/conf
- name: sentinel-conf
@@ -278,7 +278,7 @@ spec:
name: redis-sentinel
volumeMounts:
- name: data
- mountPath: /data
+ mountPath: {{ .Values.dataMountPath }}
- name: redis-config
mountPath: /etc/conf
- name: sentinel-conf
diff --git a/deploy/redis/values.yaml b/deploy/redis/values.yaml
index ca41d6f8d52..636cdfe7f56 100644
--- a/deploy/redis/values.yaml
+++ b/deploy/redis/values.yaml
@@ -17,6 +17,7 @@ imagePullSecrets: []
nameOverride: ""
fullnameOverride: ""
clusterVersionOverride: ""
+dataMountPath: /data
logConfigs:
running: /data/running.log
diff --git a/deploy/weaviate/templates/backuppolicytemplate.yaml b/deploy/weaviate/templates/backuppolicytemplate.yaml
index 579493cead8..ab7dc465956 100644
--- a/deploy/weaviate/templates/backuppolicytemplate.yaml
+++ b/deploy/weaviate/templates/backuppolicytemplate.yaml
@@ -9,14 +9,14 @@ spec:
clusterDefinitionRef: weaviate
backupPolicies:
- componentDefRef: weaviate
- retention:
- ttl: 7d
- schedule:
- snapshot:
- enable: false
- cronExpression: "0 18 * * 0"
- snapshot:
- target:
- connectionCredentialKey:
- passwordKey: password
- usernameKey: username
\ No newline at end of file
+ retentionPeriod: 7d
+ backupMethods:
+ - name: volume-snapshot
+ snapshotVolumes: true
+ targetVolumes:
+ volumes:
+ - data
+ schedules:
+ - backupMethod: volume-snapshot
+ enabled: false
+ cronExpression: "0 18 * * 0"
\ No newline at end of file
diff --git a/docker/Dockerfile b/docker/Dockerfile
index 284aecb43e4..7800c67aa68 100644
--- a/docker/Dockerfile
+++ b/docker/Dockerfile
@@ -1,7 +1,7 @@
# Build the manager binary
ARG DIST_IMG=gcr.io/distroless/static:nonroot
-ARG GO_VERSION=1.20
+ARG GO_VERSION=1.21
FROM --platform=${BUILDPLATFORM} golang:${GO_VERSION} as builder
@@ -26,12 +26,11 @@ ENV GOPROXY=${GOPROXY}
WORKDIR /src
# Copy the Go Modules manifests
-#COPY go.mod go.mod
-#COPY go.sum go.sum
+COPY go.mod go.mod
+COPY go.sum go.sum
# cache deps before building and copying source so that we don't need to re-download as much
# and so that source changes don't invalidate our downloaded layer
-RUN --mount=type=bind,target=. \
- --mount=type=cache,target=/go/pkg/mod \
+RUN --mount=type=cache,target=/go/pkg/mod \
go mod download
# Copy the go source
diff --git a/docker/Dockerfile-dataprotection b/docker/Dockerfile-dataprotection
index d06208bfa3d..45b6349d545 100644
--- a/docker/Dockerfile-dataprotection
+++ b/docker/Dockerfile-dataprotection
@@ -1,7 +1,7 @@
# Build the dataprotection binary
ARG DIST_IMG=gcr.io/distroless/static:nonroot
-ARG GO_VERSION=1.20
+ARG GO_VERSION=1.21
FROM --platform=${BUILDPLATFORM} golang:${GO_VERSION} as builder
@@ -26,12 +26,11 @@ ENV GOPROXY=${GOPROXY}
WORKDIR /src
# Copy the Go Modules manifests
-#COPY go.mod go.mod
-#COPY go.sum go.sum
+COPY go.mod go.mod
+COPY go.sum go.sum
# cache deps before building and copying source so that we don't need to re-download as much
# and so that source changes don't invalidate our downloaded layer
-RUN --mount=type=bind,target=. \
- --mount=type=cache,target=/go/pkg/mod \
+RUN --mount=type=cache,target=/go/pkg/mod \
go mod download
# Copy the go source
diff --git a/docker/Dockerfile-datascript b/docker/Dockerfile-datascript
index 6dde6ebf653..da47d704d20 100644
--- a/docker/Dockerfile-datascript
+++ b/docker/Dockerfile-datascript
@@ -1,6 +1,6 @@
# Build client images for mysql and postgres to support datascripts
-# Use alpine with tag 20230329 is corresponding to "edge" tag (latest release to date is 3.18) as of 20230625
-FROM docker.io/alpine:edge as dist
+# The latest release to date is 3.18) as of 20230625
+FROM docker.io/alpine:3.18 as dist
# ARG APK_MIRROR
# install tools via apk
@@ -12,6 +12,8 @@ RUN apk add --no-cache jq --allow-untrusted
RUN apk add --no-cache postgresql-client --allow-untrusted
RUN apk add --no-cache mysql-client mariadb-connector-c --allow-untrusted
+
+RUN apk add redis
RUN rm -rf /var/cache/apk/*
USER 65532:65532
diff --git a/docker/Dockerfile-dev b/docker/Dockerfile-dev
index 8ed81f28efe..b4dd8dda53d 100644
--- a/docker/Dockerfile-dev
+++ b/docker/Dockerfile-dev
@@ -1,7 +1,7 @@
# Based on https://github.com/microsoft/vscode-dev-containers/tree/v0.245.2/containers/go/.devcontainer/base.Dockerfile
# [Choice] Go version: 1, 1.19, 1.18, etc
-ARG GOVERSION=1.20
+ARG GOVERSION=1.21
FROM golang:${GOVERSION}-bullseye
# Copy library scripts to execute
diff --git a/docker/Dockerfile-probe b/docker/Dockerfile-probe
index 1e530e20f36..56dffc56e35 100644
--- a/docker/Dockerfile-probe
+++ b/docker/Dockerfile-probe
@@ -11,7 +11,7 @@
#TARGETARCH - Architecture from --platform, e.g. arm64
#TARGETVARIANT - used to set target ARM variant, e.g. v7
-ARG GO_VERSION=1.20
+ARG GO_VERSION=1.21
FROM --platform=${BUILDPLATFORM} golang:${GO_VERSION} as builder
ARG TARGETOS
diff --git a/docker/Dockerfile-tools b/docker/Dockerfile-tools
index 2184145d54b..ea7f76feb7b 100644
--- a/docker/Dockerfile-tools
+++ b/docker/Dockerfile-tools
@@ -11,7 +11,7 @@
#TARGETARCH - Architecture from --platform, e.g. arm64
#TARGETVARIANT - used to set target ARM variant, e.g. v7
-ARG GO_VERSION=1.20
+ARG GO_VERSION=1.21
FROM --platform=${BUILDPLATFORM} golang:${GO_VERSION} as builder
ARG TARGETOS
@@ -28,8 +28,8 @@ ENV GOPROXY=${GOPROXY}
WORKDIR /src
# Copy the Go Modules manifests
-#COPY go.mod go.mod
-#COPY go.sum go.sum
+COPY go.mod go.mod
+COPY go.sum go.sum
# cache deps before building and copying source so that we don't need to re-download as much
# and so that source changes don't invalidate our downloaded layer
# RUN go mod download
@@ -44,8 +44,7 @@ WORKDIR /src
#COPY cmd/cli/ cmd/cli/
#COPY apis/ apis/
#COPY test/testdata/testdata.go test/testdata/testdata.go
-RUN --mount=type=bind,target=. \
- --mount=type=cache,target=/go/pkg/mod \
+RUN --mount=type=cache,target=/go/pkg/mod \
go mod download
# Build
diff --git a/docs/developer_docs/integration/environment-variables-and-placeholders.md b/docs/developer_docs/integration/environment-variables-and-placeholders.md
new file mode 100644
index 00000000000..b4e6a591174
--- /dev/null
+++ b/docs/developer_docs/integration/environment-variables-and-placeholders.md
@@ -0,0 +1,57 @@
+---
+title: Environment variables and placeholders
+description: KubeBlocks Environment Variables and Placeholders
+keywords: [environment variables, placeholders]
+sidebar_position: 10
+sidebar_label: Environment variables and placeholders
+---
+
+# Environment variables and placeholders
+
+## Environment variables
+
+### Automatic pod's container environment variables
+
+The following variables are injected by KubeBlocks into each pod.
+
+| Name | Description |
+| :--- | :---------- |
+| KB_POD_NAME | K8s Pod Name |
+| KB_NAMESPACE | K8s Pod Namespace |
+| KB_SA_NAME | KubeBlocks Service Account Name |
+| KB_NODENAME | K8s Node Name |
+| KB_HOSTIP | K8s Host IP address |
+| KB_PODIP | K8s Pod IP address |
+| KB_PODIPS | K8s Pod IP addresses |
+| KB_POD_UID | POD UID (`pod.metadata.uid`) |
+| KB_CLUSTER_NAME | KubeBlocks Cluster API object name |
+| KB_COMP_NAME | Running pod's KubeBlocks Cluster API object's `.spec.components.name` |
+| KB_CLUSTER_COMP_NAME | Running pod's KubeBlocks Cluster API object's `<.metadata.name>-<.spec.components.name>` |
+| KB_REPLICA_COUNT | Running pod's component's replica |
+| KB_CLUSTER_UID | Running pods' KubeBlocks Cluster API object's `metadata.uid` |
+| KB_CLUSTER_UID_POSTFIX_8 | Last eight digits of KB_CLUSTER_UID |
+| KB_{ordinal}_HOSTNAME | Running pod's hostname, where `{ordinal}` is the ordinal of pod.
N/A if workloadType=Stateless. |
+| KB_POD_FQDN | Running pod's fully qualified domain name (FQDN).
N/A if workloadType=Stateless. |
+
+## Built-in Place-holders
+
+### ComponentValueFrom API
+
+| Name | Description |
+| :--- | :---------- |
+| POD_ORDINAL | Pod ordinal |
+| POD_FQDN | Pod FQDN (fully qualified domain name) |
+| POD_NAME | Pod Name |
+
+### ConnectionCredential API
+
+| Name | Description |
+| :--- | :---------- |
+| UUID | Generate a random UUID v4 string. |
+| UUID_B64 | Generate a random UUID v4 BASE64 encoded string. |
+| UUID_STR_B64 | Generate a random UUID v4 string then BASE64 encoded. |
+| UUID_HEX | Generate a random UUID v4 HEX representation. |
+| HEADLESS_SVC_FQDN | Headless service FQDN placeholder, value pattern - `$(CLUSTER_NAME)-$(1ST_COMP_NAME)-headless.$(NAMESPACE).svc`, where 1ST_COMP_NAME is the 1st component that provide `ClusterDefinition.spec.componentDefs[].service` attribute; |
+| SVC_FQDN | Service FQDN placeholder, value pattern - `$(CLUSTER_NAME)-$(1ST_COMP_NAME).$(NAMESPACE).svc`, where 1ST_COMP_NAME is the 1st component that provide `ClusterDefinition.spec.componentDefs[].service` attribute; |
+| SVC_PORT_{PORT_NAME} | A ServicePort's port value with specified port name, i.e, a servicePort JSON struct:
`{"name": "mysql", "targetPort": "mysqlContainerPort", "port": 3306}`, and "$(SVC_PORT_mysql)" in the connection credential value is 3306. |
+| RANDOM_PASSWD | Random 8 characters |
diff --git a/docs/developer_docs/integration/how-to-add-an-add-on.md b/docs/developer_docs/integration/how-to-add-an-add-on.md
index 37eb917bc40..dbc1842a8ec 100644
--- a/docs/developer_docs/integration/how-to-add-an-add-on.md
+++ b/docs/developer_docs/integration/how-to-add-an-add-on.md
@@ -16,7 +16,7 @@ There are altogether 3 steps to integrate an add-on:
2. Prepare cluster templates.
3. Add an `addon.yaml` file.
-## Step 1. Design a blueprint for cluster.
+## Step 1. Design a blueprint for cluster
Before getting started, make sure to design your cluster blueprint. Think about what you want your cluster to look like. For example:
@@ -37,9 +37,9 @@ Cluster Format: Deploying a MySQL 8.0 Standalone.
| ClusterVersion | Image: docker.io/mysql:8.0.34 |
| Cluster.yaml | Specified by the user during creation |
-## Step 2. Prepare cluster templates.
+## Step 2. Prepare cluster templates
-### 2.1 Create a Helm chart.
+### 2.1 Create a Helm chart
Opt 1.`helm create oracle-mysql`
@@ -226,7 +226,7 @@ Now you've finished with ClusterDefinition and ClusterVersion, try to do a quick
:::
-### 2.2 Install Helm chart.
+### 2.2 Install Helm chart
Install Helm.
@@ -245,7 +245,7 @@ REVISION: 1
TEST SUITE: None
```
-### 2.3 Create a cluster.
+### 2.3 Create a cluster
Create a MySQL cluster with `kbcli cluster create`.
@@ -313,7 +313,7 @@ After the creating, you can:
kbcli cluster stop mycluster
```
-## Step 3. Add an addon.yaml file.
+## Step 3. Add an addon.yaml file
This is the last step to integrate an add-on to KubeBlocks. After creating this addon.yaml file, this add-on is in the KubeBlocks add-on family. Please refer to `tutorial-1-create-an-addon/oracle-mysql-addon.yaml`.
diff --git a/docs/developer_docs/integration/monitoring.md b/docs/developer_docs/integration/monitoring.md
new file mode 100644
index 00000000000..464629df653
--- /dev/null
+++ b/docs/developer_docs/integration/monitoring.md
@@ -0,0 +1,267 @@
+---
+title: Monitoring
+description: How to configure monitoring function in KubeBlocks
+keywords: [monitoring]
+sidebar_position: 6
+sidebar_label: Monitoring
+---
+
+# Configure monitoring
+
+This tutorial takes Oracle MySQL as an example and explains how to configure monitoring in KubeBlocks. You can refer to [the full PR](https://github.com/apecloud/learn-kubeblocks-addon/tree/main/tutorial-4-monitor-cluster/oracle-mysql).
+
+## Before you start
+
+1. Knowledge about basic Kubernetes concepts, such as Pod and Sidecar.
+2. Finish [Tutorial 1](./how-to-add-an-add-on.md).
+3. Knowledge about basic monitoring system concepts, such as Prometheus and Grafana.
+
+## Introduction
+
+Monitoring is an essential part of Kubernetes observability. It helps developers check the system's operational status to quickly identify issues.
+
+Kubeblocks currently integrates Prometheus and Grafana as add-ons. In this tutorial, you will learn how to integrate the Prometheus/Grafana solution.
+
+### Prometheus Overview
+
+Prometheus provides an open-source monitoring solution that integrates metric collection, metric storage, and alert capabilities.
+
+It is widely used in cloud-native, containerized, and microservices architectures. With Prometheus, developers and operations teams can monitor the performance and health status of applications in real-time, so as to quickly identify and resolve issues to ensure application reliability and availability. Prometheus is usually used with Grafana to create powerful monitoring and observability solutions.
+
+### Grafana Overview
+
+Grafana is an open-source analytics and monitoring platform widely used for visualizing time series data. It allows users to create interactive and customizable dashboards to monitor and analyze data from various sources.
+
+:paperclip: Table 1. Terminology
+
+| Term | Description |
+| :-- | :---------- |
+| Prometheus Exporter | Prometheus Exporter is a component that collects monitoring data and provides data to external entities using the Prometheus monitoring specification.
For more details, refer to [Prometheus Exporter List](https://prometheus.io/docs/instrumenting/exporters/). |
+| Prometheus Metrics | Prometheus Metrics are data points used for monitoring and performance analysis. They typically include request counts, response times, CPU usage, and memory usage. |
+| Grafana Dashboard | Grafana Dashboard is a visualization interface used to present data. It is commonly used for monitoring and analyzing various time series data. |
+
+Prometheus in KubeBlocks has already been configured with scraping jobs, so developers only need to configure the Exporter. In KubeBlocks, the Exporter is deployed as a sidecar alongside the main container of the database engine in the same Pod.
+
+## Configure Exporter
+
+First, choose an Exporter. This tutorial is based on Oracle MySQL, so a MySQL Exporter is needed.
+
+### Configure Exporter version
+
+Modify ClusterVersion (`clusterversion.yaml`).
+
+You can find an appropriate Exporter from open-source communities(e.g., [Prometheus in Docker](https://hub.docker.com/u/prom)).
+
+```yaml
+componentVersions:
+- componentDefRef: mysql-compdef
+ versionsContext:
+ containers:
+ - name: mysql-container
+ image: ...
+ imagePullPolicy: ..
+ - name: mysql-exporter
+ image: prom/mysqld-exporter:v0.14.0
+```
+
+Specify the image of mysql-exporter as prom/mysqld-exporter with the version 0.14.0.
+
+### Add an Exporter container
+
+Modify `clusterdefinition.yaml` and configure mysql-exporter in Sidecar form.
+
+```yaml
+podSpec:
+ containers:
+ # mysql container and other containers ->
+ - name: mysql-exporter
+ ports:
+ - name: metrics
+ containerPort: 9104
+ protocol: TCP
+ env:
+ - name: "MYSQL_MONITOR_USER"
+ valueFrom:
+ secretKeyRef:
+ name: $(CONN_CREDENTIAL_SECRET_NAME)
+ key: username
+ - name: "MYSQL_MONITOR_PASSWORD"
+ valueFrom:
+ secretKeyRef:
+ name: $(CONN_CREDENTIAL_SECRET_NAME)
+ key: password
+ - name: "DATA_SOURCE_NAME"
+ value: "$(MYSQL_MONITOR_USER):$(MYSQL_MONITOR_PASSWORD)@(localhost:3306)/"
+```
+
+As shown from Line 4 to 21, a new container has been added to the original PodSpec. `DATA_SOURCE_NAME` is an environment variable specific to the mysql-exporter.
+
+:::caution
+
+Different Exporters require different environment variables, and they need to be configured based on specific features of each Exporter.
+
+As mentioned, this tutorial uses mysql exporter 0.14.0. In the latest mysql exporter version 0.15.0, the variable `DATA_SOURCE_NAME` is no longer supported.
+
+:::
+
+### Configure monitor parameters
+
+Modify `clusterdefinition.yaml` and configure `monitor` parameters.
+
+```yaml
+componentDefs:
+ - name: mysql-compdef
+ characterType: mysql
+ service: ..
+ monitor:
+ exporterConfig:
+ scrapePort: 9104 # Listening port of the Exporter, used by Prometheus to pull data
+ scrapePath: "/metrics" # Path of the Exporter path, used by Prometheus to pull data
+```
+
+KubeBlocks supports multiple monitoring solutions. To use the open-source Prometheus/Grafana solution, configure the listening port and metrics path in `monitor`, which should correspond to the container-port specified in [2. Add an Exporter container](#2-add-an-exporter-container).
+
+## Configure Grafana Dashboard
+
+### Obtain Grafana Dashboard configurations
+
+Grafana Dashboard can help users monitor, analyze, and understand data in real-time. For popular databases, various dashboard configuration files (in JSON format) can be easily found.
+
+- [Official Website of Grafana](https://grafana.com/grafana/dashboards).
+- [KubeBlocks Dashboard](https://github.com/apecloud/kubeblocks-mixin).
+
+### Add to your cluster
+
+Import the downloaded JSON files on the Grafana Dashboard page or configure them in your cluster template.
+
+The latter option is more versatile, as the same configuration can be reused for any cluster generated through the template.
+
+Therefore, two files are added to the existing Helm chart.
+
+- dashboards: Save dashboard JSON files.
+- grafana: Create a ConfigMap to store the contents of dashboard JSON files.
+
+```yaml
+tree oracle-mysql
+.
+├── Chart.yaml
+├── dashboards
+│  └── mysql.json
+├── templates
+│  ├── NOTES.txt
+│  ├── _helpers.tpl
+│  ├── clusterdefinition.yaml
+│  └── clusterversion.yaml
+│  └── grafana
+│  └── configmap-dashboards.yaml
+└── values.yaml
+
+4 directories, 8 files
+```
+
+## Monitor cluster data
+
+### Prepare the environment and enable Prometheus monitoring components
+
+Run `kbcli addon list` to check if the following add-ons are enabled (status: Enabled):
+
+```bash
+kbcli addon list
+>
+...
+grafana Helm Enabled true
+alertmanager-webhook-adaptor Helm Enabled true
+prometheus Helm Enabled alertmanager true
+...
+```
+
+If not (status: `Disabled`), enable them one by one.
+
+```bash
+kbcli addon enable prometheus
+kbcli addon enable alertmanager-webhook-adaptor
+kbcli addon enable grafana
+```
+
+Then you can have access to the integrated three dashboards:
+
+```bash
+kbcli dashboard list
+>
+NAME NAMESPACE PORT CREATED-TIME
+kubeblocks-grafana kb-system 13000 Jul 24,2023 11:38 UTC+0800
+kubeblocks-prometheus-alertmanager kb-system 19093 Jul 24,2023 11:38 UTC+0800
+kubeblocks-prometheus-server kb-system 19090 Jul 24,2023 11:38 UTC+0800
+```
+
+### Create a database cluster
+
+1. Install a cluster template.
+
+ ```bash
+ helm install oracle-mysql ./path-to-your-helm-chart/oracle-mysql
+ ```
+
+2. Enable monitoring function.
+
+ Opt 1. Enable when creating a cluster
+
+ ```bash
+ kbcli cluster create mycluster --cluster-definition='oracle-mysql' --monitor='true'
+ ```
+
+ Opt 2. Enable in an existing cluster
+
+ ```bash
+ kbcli cluster update mycluster --monitor='true'
+ ```
+
+3. Open the dashboard
+
+ ```bash
+ # View available dashboards
+ kbcli dashboard list
+ >
+ NAME NAMESPACE PORT CREATED-TIME
+ kubeblocks-grafana default 3000 Jan 13,2023 10:53 UTC+0800
+ kubeblocks-prometheus-alertmanager default 9093 Jan 13,2023 10:53 UTC+0800
+ kubeblocks-prometheus-server default 9090 Jan 13,2023 10:53 UTC+0800
+
+ # Select Grafana and open the web console in the default browser
+ kbcli dashboard open kubeblocks-grafana
+ ```
+
+4. Sign in to the dashboard
+
+![Grafana Homepage](./../../img/addon-monitoring-signin.png)
+
+:::note
+
+If the dashboard requires a login, use the following username and password.
+
+```bash
+Username: admin
+Password: kubeblocks
+```
+
+:::
+
+## Summary
+
+This tutorial explains how to quickly adapt the Prometheus/Grafana solution to monitor your database cluster. KubeBlocks will also introduce a monitoring solution based on OpenTelemetry in the future. Stay tuned for updates.
+
+## References
+
+1. [Prometheus](https://prometheus.io/).
+2. [Grafana Dashboard](https://grafana.com/grafana/dashboards/).
+3. [Create a dashboard](https://grafana.com/docs/grafana/latest/dashboards/build-dashboards/create-dashboard/).
+
+## Appendix
+
+### A.1 Disable cluster monitoring
+
+In KubeBlocks, you can enable or disable monitoring for a specific cluster using kbcli.
+
+```bash
+kbcli cluster update mycluster --monitor='false'
+```
diff --git a/docs/developer_docs/integration/multi-component.md b/docs/developer_docs/integration/multi-component.md
new file mode 100644
index 00000000000..ef0c497f584
--- /dev/null
+++ b/docs/developer_docs/integration/multi-component.md
@@ -0,0 +1,254 @@
+---
+title: Multi-component configuration
+description: How to configure multi-component in KubeBlocks with NebulaGraph as an example
+keywords: [multi-component,add-on]
+sidebar_position: 7
+sidebar_label: Multi-component configuration
+---
+
+# Multi-component configuration
+
+So far, you've learned the definition, backup, and configuration of single-component clusters (e.g., Oracle-MySQL).
+
+This tutorial takes NebulaGraph as an example to demonstrate how to integrate a multi-component cluster and address several common issues in multi-component configurations. You can find more details in [this repository](https://github.com/apecloud/kubeblocks/tree/main/deploy/nebula).
+
+## Before you start
+
+- Finish [Tutorial 1](./how-to-add-an-add-on.md).
+- Knowledge about basic KubeBlocks concepts, such as ClusterDefinition, Cluster, ComponentRef, and Component.
+
+## NebulaGraph Architecture
+
+First, take a look at the overall architecture of NebulaGraph.
+
+NebulaGraph applies the separation of storage and computing architecture and consists of three services: the Graph Service, the Meta Service, and the Storage Service. The following figure shows the architecture of a typical NebulaGraph cluster.
+
+![NebulaGraph Architecture (source: https://github.com/vesoft-inc/nebula)](./../../img/nebula-aichitecture.png)
+
+- Metad: It is a component based on the Raft protocol and is responsible for data management tasks such as Schema operations, cluster management, and user permission management.
+- Graphd: It is the compute component and is responsible for handling query requests, including query parsing, validation, and generating and executing query plans.
+- Storaged: It is the distributed storage component based on Multi Group Raft, responsible for storing data.
+
+If the client is considered, the fourth component is:
+
+- Client: It is a stateless component used to connect to Graphd and send graph queries.
+
+## Configure cluster typology
+
+Now you've learned the four components of NebulaGraph, and how each component is started and configured.
+
+Similar to a single-component cluster, you can quickly assemble the definition for a multi-component cluster.
+
+```yaml
+apiVersion: apps.kubeblocks.io/v1alpha1
+kind: ClusterDefinition
+metadata:
+ name: nebula
+spec:
+ componentDefs:
+ - name: nebula-console # client
+ workloadType: Stateless
+ characterType: nebula
+ podSpec: ...
+ - name: nebula-graphd # graphd
+ workloadType: Stateful
+ podSpec: ...
+ - name: nebula-metad # metad
+ workloadType: Stateful
+ podSpec: ...
+ - name: nebula-storaged # storaged
+ workloadType: Stateful
+ podSpec: ...
+---
+apiVersion: apps.kubeblocks.io/v1alpha1
+kind: ClusterVersion
+metadata:
+ name: nebula-3.5.0
+spec:
+ clusterDefinitionRef: nebula # clusterdef name
+ componentVersions:
+ - componentDefRef: nebula-console # Specify image for client
+ versionsContext:
+ containers:
+ - name: nebula-console
+ image: ...
+ - componentDefRef: nebula-graphd # Specify image for graphd
+ versionsContext:
+ containers:
+ - name: nebula-graphd
+ image:
+ - componentDefRef: nebula-metad # Specify image for metad
+ versionsContext:
+ containers:
+ - name: nebula-metad
+ image: ...
+ - componentDefRef: nebula-storaged # Specify image for storaged
+ versionsContext:
+ containers:
+ - name: nebula-storaged
+ image: ...
+```
+
+The above YAML file provides an outline of the ClusterDefinition and ClusterVersion for NebulaGraph. Corresponding to Figure 1., four components (including the client) and their version information are specified.
+
+If each component can be started independently, the information provided in Figure 2. would be sufficient.
+
+However, it can be observed that in a multi-component cluster, there are often inter-component references. So, how to specify the references thereof?
+
+## Configure inter-component references
+
+As discovered, components may refer to each other and Figure 3. shows the inter-component references in a NebulaGraph cluster. For example,
+
+1. Nebula-Console needs to know the port number and service name of Nebula-Graphd.
+2. Nebula-Graphd needs to know the DNS of each Pod of Nebula-Metad.
+3. Nebula-Storaged also needs to know the DNS of each Pod of Nebula-Metad.
+
+![Nebula Inter-Component References](./../../img/nebula-inter-component-ref.png)
+
+Therefore, three common types of inter-component references are: \
+
+1. **Service Reference**
+ e.g., Nebula-Console needs to obtain the service name of Nebula-Graphd.
+2. **HostName Reference**
+ e.g., Nebula-Graphd needs to configure the DNS of all Pods of Nebula-metad. This reference typically points to a stateful component.
+3. **Field Reference**
+ e.g., Nebula-Console needs to obtain a service port name of Nebula-Graphd.
+
+To ensure that the cluster starts normally, the above information needs to be injected into the Pod through environment variables (whether it is loaded through configmap or defined as pod env).
+
+In KubeBlocks, the `ComponentDefRef` API can be used to achieve the goal. It introduces the following APIs:
+
+- `componentDefName`, used to specify the name of the component definition that is being referenced to.
+- `componentRefEnv`, which defines a set of environment variables that need to be injected.
+ - `name` defines the name of the injected environment variable.
+ - `valueFrom` defines the source of the variable value.
+
+Next, you will learn how `ComponentDefRef` deals with the three types of references mentioned above.
+
+### Service Reference
+
+Case 1: Nebula-Console needs to obtain the service name of Nebula-Graphd.
+
+When defining `nebula-console`, add the following definitions (as `componentDefRef` shows):
+
+```yaml
+ - name: nebula-console
+ workloadType: Stateless
+ characterType: nebula
+ componentDefRef:
+ - componentDefName: nebula-graphd
+ componentRefEnv:
+ - name: GRAPHD_SVC_NAME
+ valueFrom:
+ type: ServiceRef
+```
+
+- Specify the component that is being referenced to as `nebula-graphd`.
+- The name of the injected environment variable is `GRAPHD_SVC_NAME`.
+- The value type of the variable is `ServerRef`, indicating that the value comes from the service name of the referenced component.
+
+:::note
+
+In KubeBlocks, if you've defined the `service` for a component, when you create a cluster, KubeBlocks will create a service named `{clusterName}-{componentName}` for that component.
+
+:::
+
+### HostName Reference
+
+Case 2: Nebula-Graphd needs to configure the DNS of all PODs of Nebula-Metad.
+
+```yaml
+ - name: nebula-graphd
+ workloadType: Statelful
+ componentDefRef:
+ - componentDefName: nebula-metad
+ componentRefEnv:
+ - name: NEBULA_METAD_SVC
+ valueFrom:
+ type: HeadlessServiceRef
+ format: $(POD_FQDN):9559 # Optional, specify value format
+```
+
+- Specify the component that is being referenced to as nebula-metad.
+- The name of the injected environment variable is NEBULA_METAD_SVC.
+- The value type of the variable is HeadlessServiceRef.
+ - It indicates that the value comes from the FQDN of all Pods of the referenced component, and multiple values are connected with , by default.
+ - If the default FQDN format does not meet your needs, customize the format through format (as shown in Line 9).
+
+:::note
+
+KubeBlocks provides three built-in variables as placeholders and they will be replaced with specific values when the cluster is created:
+- ${POD_ORDINAL}, which is the ordinal number of the Pod.
+- ${POD_NAME}, which is the name of the Pod, formatted as `{clusterName}-{componentName}-{podOrdinal}`.
+- ${POD_FQDN}, which is the Fully Qualified Domain Name (FQDN) of the Pod.
+
+In KubeBlocks, each stateful component has a Headless Service named `headlessServiceName = {clusterName}-{componentName}-headless` by default.
+
+Therefore, the format of the Pod FQDN of each stateful component is:
+`POD_FQDN = {clusterName}-{componentName}-{podIndex}.{headlessServiceName}.{namespace}.svc`.
+
+:::
+
+### Field Reference
+
+Case 3: Nebula-Console needs to obtain a service port name of Nebula-Graphd.
+
+When defining `nebula-console` , add the following configurations (as `componentDefRef` shows):
+
+```yaml
+ - name: nebula-console
+ workloadType: Stateless
+ characterType: nebula
+ componentDefRef:
+ - componentDefName: nebula-graphd
+ componentRefEnv:
+ - name: GRAPHD_SVC_PORT
+ valueFrom:
+ type: FieldRef
+ fieldPath: $.componentDef.service.ports[?(@.name == "thrift")].port
+```
+
+- Specify the component that is being referenced to as `nebula-graphd`.
+- The name of the injected environment variable is `GRAPHD_SVC_PORT`.
+- The value type of the variable is `FieldRef`, indicating that the value comes from a certain property value of the referenced component and is specified by `fieldPath`.
+
+`fieldPath` provides a way to parse property values through JSONPath syntax.
+When parsing JSONPath, KubeBlocks registers two root objects by default:
+
+- **componentDef**, the componentDef object being referenced.
+- **components**, all components corresponding to the componentDef in the created cluster.
+
+Therefore, in `fieldPath`, you can use `$.componentDef.service.ports[?(@.name == "thrift")].port` to obtain the port number named `thrift` in the service defined by this component.
+
+## Summary
+
+This tutorial takes NebulaGraph as an example and introduces several types and solutions of inter-component references.
+
+In addition to NebulaGraph, engines like GreptimDB, Pulsar, RisingWave and StarRocks also adopt `componentDefRef` API to deal with component references. You can also refer to their solutions.
+
+For more information about the `componentDefRef`, refer to [ComponentDefRef API](https://kubeblocks.io/docs/release-0.6/developer_docs/api-reference/cluster#apps.kubeblocks.io/v1alpha1.ComponentDefRef).
+
+## Appendix
+
+### A1. YAML tips
+
+Since Nebula-Graphd, Nebula-Metad and Nebula-Storaged all require the FQDN of each Pod in Nebula-Metad, you don't need to configure them repeatedly.
+
+Quickly configure them with YAML anchors.
+
+```yaml
+- name: nebula-graphd
+ # ...
+ componentDefRef:
+ - &metadRef # Define an anchor with `&`
+ componentDefName: nebula-metad
+ componentRefEnv:
+ - name: NEBULA_METAD_SVC
+ valueFrom:
+ type: HeadlessServiceRef
+ format: $(POD_FQDN){{ .Values.clusterDomain }}:9559
+ joinWith: ","
+- name: nebula-storaged
+ componentDefRef:
+ - *metadRef # Use the anchor with `*` to avoid duplication
+```
diff --git a/docs/developer_docs/integration/parameter-configuration.md b/docs/developer_docs/integration/parameter-configuration.md
new file mode 100644
index 00000000000..d56cb7b2989
--- /dev/null
+++ b/docs/developer_docs/integration/parameter-configuration.md
@@ -0,0 +1,353 @@
+---
+title: Parameter configuration
+description: How to configure parameter templates and update parameters in KubeBlocks
+keywords: [parameter configuration]
+sidebar_position: 5
+sidebar_label: Parameter configuration
+---
+
+# Parameter configuration
+
+This tutorial takes Oracle MySQL as an example and explains how to configure parameter templates and parameters in KubeBlocks. You can find [the full PR here](https://github.com/apecloud/learn-kubeblocks-addon/tree/main/tutorial-3-config-and-reconfig/).
+
+## Before you start
+
+1. Grasp basic concepts of Kubernetes, such as Pod and ConfigMap.
+2. Finish configurations in [Configure parameter template](./parameter-template.md).
+3. (Optional) Know something about Go Template.
+4. (Optional)Know something about CUE Lang.
+
+## Introduction
+
+KubeBlocks adds configurations by mounting the ConfigMap to the volume. With a Kubernetes-Native concept that `ConfigMap is the only source of truth`, it centralizes entry for parameter changes in the ConfigMap to prevent configuration drifting. Therefore, the order below illustrates how KubeBlocks performs parameter reconfiguration:
+
+1. Configure parameter values in the ConfigMap.
+2. Derive parameter configurations (add/delete/update) based on ConfigMap modifications.
+3. Apply the parameter configurations to the engine.
+
+Different parameters require different configuration methods:
+
+- Static parameters require a cluster restart (cold update).
+- Dynamic parameters require a parameter refresh (hot update).
+
+Table 1 lists four common hot update methods, including UNIX Signal, SQL, Auto, etc. Currently, engines in KubeBlocks can implement one or more of these methods. For example, to apply dynamic configuration in PostgreSQL, you can use:
+
+- UNIX Signal: Send a `SIGHUP` signal.
+- Tools: Call `pg_ctl` command.
+- SQL: Execute SQL statements to directly update parameters.
+
+:paperclip: Table 1. Summary of Parameter Hot Updates
+
+| Methods | Descriptions | Applicability |
+| :---------- | :----------- | :------------ |
+| Unix Signal | For example, PostgreSQL.
If you need to reload the configuration file after parameter configuration, send a `SIGHUP` signal to PG. | Applicable to engines that support Unix Signal updates. |
+| SQL | For example, MySQL.
Perform parameter configurations through the SQL statement `SET GLOBAL =`. | Applicable to most RDBMS engines.
**Note**: The `execSQL` interface is required. Currently, KubeBlocks only supports MySQL and PostgreSQL. |
+| Tools | For example, Redis or MongoDB.
Related tools are provided for configuring parameters. | Implemented via custom scripts or local tools, highly versatile. |
+| Auto | The engine itself watches for changes in configuration files, and updates automatically when a change is detected. | Dependent on whether the engine supports automatic loading. |
+
+As mentioned in [Parameter template](./parameter-template.md), Kubernetes does not synchronously update ConfigMap changes to the Pod. For KubeBlocks, it not only needs to distinguish the way parameters are configured but also needs to watch whether the corresponding configurations are synchronized to the Pod.
+
+Now take a look at how KubeBlocks manages parameter configurations through the `ConfigConstraint` API.
+
+## ConfigConstraint
+
+As a multi-engine platform, KubeBlocks needs to get the following information to better support parameter configuration:
+
+1. Format of configuration files:
+
+ Different configuration files have different structures. KubeBlocks parses files based on their structure to deduce the information about each configuration (add/delete/update).
+
+2. Effect scope of parameters:
+
+ Be clear which parameters are dynamic, which are static, and which are immutable. KubeBlocks specifies the effect scope of parameters to determine how they quickly take effect.
+
+3. Methods for dynamic parameter changes:
+
+ As shown in Table 1., parameters can be dynamically configured in various ways. Therefore, specify different dynamic configuration methods for different engines.
+
+4. Definition of parameter validation rules:
+
+ It's important to define validation rules for parameters. In a production environment, developers often fail to start databases due to typos in parameter values. Parameter validation therefore adds a layer of protection by performing checks in advance to prevent such mistakes.
+
+:::note
+
+KubeBlocks creates a config-manager sidecar for components that have configured ConfigConstraint. It is used to detect file updates, send signals, and execute updated scripts.
+
+:::
+
+The information is included in `ConfigConstraint` (parameter constraints) as shown below. The four sections correspond to the four key configuration details mentioned above.
+
+```yaml
+apiVersion: apps.kubeblocks.io/v1alpha1
+kind: ConfigConstraint
+metadata:
+ name: oracle-mysql-config-constraints
+spec:
+ #1. Specify the file format as INI and only focus on the `mysqld` section
+ formatterConfig:
+ format: ini
+ iniConfig:
+ sectionName: mysqld
+
+ #2. Specify the dynamic parameter configuration method for MySQL, using `reload-script` to execute SQL statements
+ reloadOptions:
+ tplScriptTrigger:
+ sync: true
+ # Specify which script file to use for configuration
+ scriptConfigMapRef: oracle-mysql-reload-script
+ namespace: {{ .Release.Namespace }}
+
+ ##3.1 Configure static parameters
+ staticParameters:
+ - open_files_limit
+ - performance_schema
+ - enforce_gtid_consistency
+
+ ##3.2 Configure dynamic parameters
+ dynamicParameters:
+ - innodb_buffer_pool_size
+ - max_connections
+ - gtid_mode
+
+ ##4. Define parameter validation rules with a CUE template
+ cfgSchemaTopLevelName: MysqlParameter
+ configurationSchema:
+ cue: |-
+ {{- .Files.Get "config/oracle-mysql-config-constraint.cue" | nindent 6 }}
+```
+
+Each API is to be explained in the following tutorial.
+
+### FormatterConfig
+
+FormatterConfig describes the configuration file format, such as `ini`, `yaml`, `json`, `xml`, `properties`.
+
+The file itself is just a text and requires different parsers.
+
+When KubeBlocks detects a configuration file change, it deduces the parameter configuration (add/delete/update) based on the format and notifies the Pod to update.
+
+For example, MySQL's adjustable parameters take the `ini` format and only parse the `mysqld` information.
+
+```bash
+ formatterConfig:
+ format: ini # Format of the configuration file and ini, xml, yaml, json and hcl are supported
+ iniConfig:
+ sectionName: mysqld # If the ini format is adopted, there might be multiple sections and sectionName is required
+```
+
+### ReloadOptions
+
+ReloadOptions describes the method of dynamic parameter configuration.
+
+Table 1 above summarizes 4 common methods of dynamic parameter configuration. KubeBlocks, accordingly, supports multiple configuration methods.
+
+- tplScriptTrigger: Configures parameter by template files.
+- shellTrigger: Configures parameters by executing scripts.
+- unixSignalTrigger: Configures parameters through UNIX Signal.
+- None: AutoLoad mode, which is automatically configured by the database engine.
+
+***Example***
+
+- tplScriptTrigger
+
+ This example chooses `tplScriptTrigger` to configure parameters by defining the content in the template file.
+
+ ```bash
+ reloadOptions:
+ tplScriptTrigger: # Configure parameters by template file
+ sync: true # Synchronou reloading
+ scriptConfigMapRef: oracle-mysql-reload-script # The referenced template file
+ namespace: {{ .Release.Namespace }}
+ ```
+
+- shellTrigger
+
+ `shellTrigger` performs dynamic parameter configuration by shell scripts, which is a general method since most databases support configuring parameters through clients.
+
+ ```yaml
+ reloadOptions:
+ shellTrigger:
+ sync: true
+ command:
+ - "update-dynamic-config.sh"
+ ```
+
+:::note
+
+The scripts in Reloadptions will be loaded to the Pod and executed by the config-manager sidecar mentioned before.
+
+:::
+
+### Static/Dynamic Parameters
+
+KubeBlocks supports configuring dynamic, static and immutable parameters and such effect scope is used to identify the parameter type and to determine how the parameter reconfiguration takes effect.
+
+KubeBlocks includes multiple parameter reloading strategies and applies the appropriate strategy based on the reconfiguration contents.
+
+This example lists some common MySQL parameters, such as the static parameter `performance_schema` and the dynamic parameter `max_connection`.
+
+If the parameter list is too long, it is recommended to use the `.Files.Get` function.
+
+```yaml
+ ##3.1 Configure static parameter list
+ staticParameters:
+ - open_files_limit
+ - performance_schema
+ - enforce_gtid_consistency
+
+ ##3.2 Configure dynamic parameter list
+ dynamicParameters:
+ - innodb_buffer_pool_size
+ - max_connections
+ - gtid_mode
+```
+
+### ConfigurationSchema
+
+During the configuration process, starting a cluster may fail due to entering an invalid parameter value.
+
+KubeBlocks provides ConfigurationSchema for validating parameter effectiveness. KubeBlocks uses CUE for verification. It works by describing the type, default value and range of each parameter to prevent problems caused by an invalid parameter value.
+
+This example illustrates the configuration for verifying MySQL parameter values.
+
+```yaml
+#MysqlParameter: {
+
+ // Sets the autocommit mode
+ autocommit?: string & "0" | "1" | "OFF" | "ON"
+
+ open_files_limit: int | *5000
+
+ // Enables or disables the Performance Schema
+ performance_schema: string & "0" | "1" | "OFF" | "ON" | *"0"
+
+ // The number of simultaneous client connections allowed.
+ max_connections?: int & >=1 & <=100000
+ ...
+ }
+```
+
+For example, the example above defines some constraints for the parameter `performance_schem` in MySQL.
+
+- Type: string
+- Available values: ON, OFF, 0, 1
+- Default value: 0
+
+```yaml
+ // Enables or disables the Performance Schema
+ performance_schema: string & "0" | "1" | "OFF" | "ON" | *"0"
+```
+
+## How to configure parameters
+
+Better user experience, KubeBlocks offers kbcli for your convenient parameter management.
+
+### Create a cluster
+
+Both kbcli and Helm are supported.
+
+
+
+
+
+```bash
+kbcli cluster create mycluster --cluster-definition='oracle-mysql' --cluster-version oracle-mysql-8.0.32
+```
+
+
+
+
+
+```bash
+helm install oracle-mysql path-to-your-helm-chart/oracle-mysql
+```
+
+
+
+
+
+### View parameter configuration
+
+View the detailed configuration of a cluster, including the configuration template name and constraint name.
+
+```bash
+kbcli cluster describe-config mycluster
+>
+ConfigSpecs Meta:
+CONFIG-SPEC-NAME FILE ENABLED TEMPLATE CONSTRAINT RENDERED COMPONENT CLUSTER
+mysql-config my.cnf true oracle-mysql-config-template oracle-mysql-config-constraints mycluster-mysql-comp-mysql-config mysql-comp mycluster
+
+History modifications:
+OPS-NAME CLUSTER COMPONENT CONFIG-SPEC-NAME FILE STATUS POLICY PROGRESS CREATED-TIME VALID-UPDATED
+```
+
+### Configure parameters
+
+For example, configure the max_connection of MySQL.
+
+Based on the above configuration,
+
+- max_connection is a dynamic parameter.
+- The value range is [1, 10000].
+
+:::note
+
+For KubeBlocks v0.6.0 and above, run `kbcli cluster edit-config` to configure the parameter.
+
+:::
+
+```bash
+kbcli cluster edit-config mycluster
+```
+
+In the interactive editing interface, edit max_connection as 1000.
+
+![Interactive Config Editor](./../../img/addon-interactive-config-editor.png)
+
+Save the changes and confirm the information to realize the parameter reconfiguration.
+
+![Confirm Config Changes](./../../img/addon-confirm-config-changes.png)
+
+### View the change history
+
+View the parameter configurations again. Besides the parameter template, the history and detailed information are also recorded.
+
+```bash
+kbcli cluster describe-config mycluster
+>
+ConfigSpecs Meta:
+CONFIG-SPEC-NAME FILE ENABLED TEMPLATE CONSTRAINT RENDERED COMPONENT CLUSTER
+mysql-config my.cnf true oracle-mysql-config-template oracle-mysql-config-constraints mycluster-mysql-comp-mysql-config mysql-comp mycluster
+
+History modifications:
+OPS-NAME CLUSTER COMPONENT CONFIG-SPEC-NAME FILE STATUS POLICY PROGRESS CREATED-TIME VALID-UPDATED
+mycluster-reconfiguring-7p442 mycluster mysql-comp mysql-config my.cnf Succeed 1/1 Aug 25,2023 18:27 UTC+0800 {"my.cnf":"{\"mysqld\":{\"max_connections\":\"1000\"}}"}
+```
+
+## Reference
+
+- [Configure a Pod to Use a ConfigMap](https://kubernetes.io/docs/tasks/configure-pod-container/configure-pod-configmap/)
+- [CUE Lang Overview](https://cuetorials.com/zh/overview/)
+- [KubeBlocks ApeCloud MySQL Configuration](https://kubeblocks.io/docs/preview/user_docs/kubeblocks-for-mysql/configuration)
+
+## Appendix
+
+### A.1 How to view the reconfiguration process
+
+Parameter configuration is a type of KubeBlocks operations, shorten as ops.
+
+After the kbcli reconfiguration command is performed, a Configuration ops is generated in KubeBlocks.
+
+As shown in Section 3.5, an ops named `mycluster-reconfiguring-7p442` is generated and you can run the command below to view the process, including the changes, policy and time.
+
+```bash
+kbcli cluster describe-op
+```
+
+### A.2 Compare the difference between two changes
+
+Run `diff-config` to view the difference between two changes
+
+```bash
+kbcli cluster diff-config
+```
diff --git a/docs/developer_docs/integration/parameter-template.md b/docs/developer_docs/integration/parameter-template.md
new file mode 100644
index 00000000000..bcea62da59e
--- /dev/null
+++ b/docs/developer_docs/integration/parameter-template.md
@@ -0,0 +1,230 @@
+---
+title: Parameter template
+description: How to configure parameter templates in KubeBlocks
+keywords: [parameter template]
+sidebar_position: 4
+sidebar_label: Parameter template
+---
+
+import Tabs from '@theme/Tabs';
+import TabItem from '@theme/TabItem';
+
+# Parameter template
+
+This tutorial demonstrates how to configure parameter templates in KubeBlocks with Oracle MySQL as an example. You can find [the full PR here](https://github.com/apecloud/learn-kubeblocks-addon/tree/main/tutorial-3-config-and-reconfig/).
+
+## Before you start
+
+1. Grasp basic concepts of Kubernetes, such as Pod and ConfigMap.
+2. Finish [Tutorial 1](./how-to-add-an-add-on.md).
+3. Know something about Go Template (Optional).
+
+## Introduction
+
+When creating a cluster, developers typically configure parameters according to resource availability, performance needs, environment, etc. Cloud database providers like AWS and Alibaba Cloud have therefore offered various parameter templates (such as high-performance and asynchronous templates for RDS) to facilitate a quick startup for users.
+
+In this tutorial, you will learn how to configure parameters in KubeBlocks, which includes adding parameter templates, configuring parameters, and configuring parameter validation.
+
+Although Kubernetes allows users to mount parameter files as ConfigMap on volumes of the Pod, it only manages ConfigMap updates and synchronizes them to the volume. Therefore, if the database engine (such as MySQL and Postgres) fails to support dynamic loading of configuration files, you can only log in to the database to perform update operations, which can easily lead to configuration drift.
+
+To prevent that, KubeBlocks manages all parameters through ConfigMap with the principle that `ConfigMap is the only source-of-truth`. It means that all parameter configurations are first applied to ConfigMap, and then, depending on different ways the parameters take effect, applied to each Pod in the cluster. A comprehensive guide on how to configure parameters will be provided in the next tutorial.
+
+## ConfigTemplate
+
+KubeBlocks renders parameter templates with ***Go Template***. Apart from common functions, it also includes some frequently-used calculation functions such as `callBufferSizeByResource` and `getContainerCPU`.
+
+With KubeBlocks's enhanced rendering capabilities, you can quickly create an ***Adaptive ConfigTemplate*** and generate appropriate configuration files based on the context, such as memory and CPU.
+
+### Add a parameter template
+
+```yaml
+1 apiVersion: v1
+2 kind: ConfigMap
+3 metadata:
+4 name: oracle-mysql-config-template
+5 labels:
+6 {{- include "oracle-mysql.labels" . | nindent 4 }}
+7 data:
+8 my.cnf: |-
+9 {{`
+10 [mysqld]
+11 port=3306
+12 {{- $phy_memory := getContainerMemory ( index $.podSpec.containers 0 ) }}
+13 {{- $pool_buffer_size := ( callBufferSizeByResource ( index $.podSpec.containers 0 ) ) }}
+14 {{- if $pool_buffer_size }}
+15 innodb_buffer_pool_size={{ $pool_buffer_size }}
+16 {{- end }}
+17
+18 # If the memory is less than 8Gi, disable performance_schema
+19 {{- if lt $phy_memory 8589934592 }}
+20 performance_schema=OFF
+21 {{- end }}
+22
+23 [client]
+24 port=3306
+25 socket=/var/run/mysqld/mysqld.sock
+26 `
+27 }}
+```
+
+The above example illustrates an adaptive ConfigTemplate for MySQL defined through ConfigMap. It includes several common MySQL parameters, such as `port` and `innodb_buffer_pool_size`.
+
+Based on the memory parameter configured when the container is started, it can:
+
+- Calculate the size of `innodb_buffer_size` (Line 11 to 15);
+- Disable `performance_schema` when the memory is less than 8Gi to reduce performance impact (Line 19 to 21).
+
+`callBufferSizeByResource` is a predefined bufferPool calculation rule, primarily for MySQL. You can also customize your calculation formulas by querying memory and CPU:
+
+- `getContainerMemory` retrieves the memory size of a particular container in the Pod.
+- `getContainerCPU` retrieves the CPU size of a particular container in the Pod.
+
+:::note
+
+Tailor additional parameter calculation options as you wish:
+
+- Calculate an appropriate `max_connection` value based on memory size.
+- Calculate reasonable configurations for other components based on the total memory available.
+
+:::
+
+### Use a parameter template
+
+#### Modify ClusterDefinition
+
+Specify parameter templates through `configSpecs` in `ClusterDefinition` and quote the ConfigMap defined in [Add a parameter template](#add-a-parameter-template).
+
+```yaml
+ componentDefs:
+ - name: mysql-compdef
+ configSpecs:
+ - name: mysql-config
+ templateRef: oracle-mysql-config-template # Defines the ConfigMap name for the parameter template
+ volumeName: configs # Name of the mounted volume
+ namespace: {{ .Release.Namespace }} # Namespace of the ConfigMap
+ podSpec:
+ containers:
+ - name: mysql-container
+ volumeMounts:
+ - mountPath: /var/lib/mysql
+ name: data
+ - mountPath: /etc/mysql/conf.d # Path to the mounted configuration files, engine-related
+ name: configs # Corresponds to the volumeName on Line 6
+ ports:
+ ...
+```
+
+As shown above, you need to modify `ClusterDefinition.yaml` file by adding `configSpecs`. Remember to specify the following:
+
+- templateRef: The name of the ConfigMap where the template is.
+- volumeName: The name of the volume mounted to the Pod.
+- namespace: The namespace of the template file (ConfigMap is namespace-scoped, usually in the namespace where KubeBlocks is installed).
+
+#### View configuration info
+
+When a new cluster is created, KubeBlocks renders the corresponding ConfigMap based on configuration templates and mounts it to the `configs` volume.
+
+1. Install a Helm chart.
+
+ ```bash
+ helm install oracle-mysql path-to-your-helm-char/oracle-mysql
+ ```
+
+2. Create a cluster.
+
+ ```bash
+ kbcli cluster create mycluster --cluster-definition oracle-mysql --cluster-version oracle-mysql-8.0.32
+ ```
+
+3. View configuration.
+
+ kbcli provides the subcommand `describe-config` to view the configuration of a cluster.
+
+ ```bash
+ kbcli cluster describe-config mycluster --component mysql-compdef
+ >
+ ConfigSpecs Meta:
+ CONFIG-SPEC-NAME FILE ENABLED TEMPLATE CONSTRAINT RENDERED COMPONENT CLUSTER
+ mysql-config my.cnf false oracle-mysql-config-template mycluster-mysql-compdef-mysql-config mysql-compdef mycluster
+
+ History modifications:
+ OPS-NAME CLUSTER COMPONENT CONFIG-SPEC-NAME FILE STATUS POLICY PROGRESS CREATED-TIME VALID-UPDATED
+ ```
+
+You can view:
+
+- Name of the configuration template: oracle-mysql-config-template
+- Rendered ConfigMap: mycluster-mysql-compdef-mysql-config
+- Name of the file loaded: my.cnf
+
+## Summary
+
+This tutorial introduces how to render "adaptive" parameters with configuration templates in KubeBlocks.
+
+In Kubernetes, ConfigMap changes are periodically synchronized to Pods, but most database engines (such as MySQL, PostgreSQL, and Redis) do not actively load new configurations. This is because modifying ConfigMap alone does not provide the capability to Reconfig (parameter changes).
+
+## Appendix
+
+### A.1 How to configure multiple parameter templates?
+
+To meet various requirements, developers often need to configure multiple parameter templates in a production environment. For example, Alibaba Cloud provides many high-performance parameter templates and asynchronous templates for customized needs.
+
+In KubeBlocks, developers can use multiple `ClusterVersion` to achieve their goals.
+
+$$Cluster = ClusterDefinition.yaml \Join ClusterVersion.yaml \Join Cluster.yaml
+$$
+
+The JoinKey is the Component Name.
+
+As the cluster definition formula indicates, multiple ClusterVersion can be combined with the same ClusterDefinition to set up different configurations.
+
+```yaml
+## The first ClusterVersion, uses configurations from ClusterDefinition
+apiVersion: apps.kubeblocks.io/v1alpha1
+kind: ClusterVersion
+metadata:
+ name: oracle-mysql
+spec:
+ clusterDefinitionRef: oracle-mysql
+ componentVersions:
+ - componentDefRef: mysql-compdef
+ versionsContext:
+ containers:
+ - name: mysql-container
+ ...
+---
+## The second ClusterDefinition, defines its own configSpecs and overrides the configuration of ClusterDefinition
+apiVersion: apps.kubeblocks.io/v1alpha1
+kind: ClusterVersion
+metadata:
+ name: oracle-mysql-perf
+spec:
+ clusterDefinitionRef: oracle-mysql
+ componentVersions:
+ - componentDefRef: mysql-compdef
+ versionsContext:
+ containers:
+ - name: mysql-container
+ ...
+ # The name needs to be consistent with that of the ConfigMap defined in ClusterDefinition
+ configSpecs:
+ - name: mysql-config
+ templateRef: oracle-mysql-perf-config-template
+ volumeName: configs
+```
+
+As shown above, two ClusterVersion objects are created.
+
+The first one uses the default parameter template (without any configuration), and the second one specifies a new parameter template `oracle-mysql-perf-config-template` through `configSpecs`.
+
+When creating a cluster, you can specify `ClusterVersion` to create clusters with different configurations, such as:
+
+```bash
+kbcli cluster create mysqlcuster --cluster-definition oracle-mysql --cluster-version oracle-mysql-perf
+```
+
+:::note
+
+KubeBlocks merges configurations from ClusterVersion and ClusterDefinition via `configSpecs.name`. Therefore, make sure that `configSpecs.name` defined in ClusterVersion matches the name defined in ClusterDefinition.
+
+:::
diff --git a/docs/img/addon-confirm-config-changes.png b/docs/img/addon-confirm-config-changes.png
new file mode 100644
index 00000000000..634c09887c4
Binary files /dev/null and b/docs/img/addon-confirm-config-changes.png differ
diff --git a/docs/img/addon-interactive-config-editor.png b/docs/img/addon-interactive-config-editor.png
new file mode 100644
index 00000000000..df5ab9df07f
Binary files /dev/null and b/docs/img/addon-interactive-config-editor.png differ
diff --git a/docs/img/addon-monitoring-signin.png b/docs/img/addon-monitoring-signin.png
new file mode 100644
index 00000000000..1cc5c4cec50
Binary files /dev/null and b/docs/img/addon-monitoring-signin.png differ
diff --git a/docs/img/nebula-aichitecture.png b/docs/img/nebula-aichitecture.png
new file mode 100644
index 00000000000..941d049f870
Binary files /dev/null and b/docs/img/nebula-aichitecture.png differ
diff --git a/docs/img/nebula-inter-component-ref.png b/docs/img/nebula-inter-component-ref.png
new file mode 100644
index 00000000000..f6733465b8c
Binary files /dev/null and b/docs/img/nebula-inter-component-ref.png differ
diff --git a/docs/user_docs/backup-and-restore/backup/backup-repo.md b/docs/user_docs/backup-and-restore/backup/backup-repo.md
index 395a496c0a9..73246bc8624 100644
--- a/docs/user_docs/backup-and-restore/backup/backup-repo.md
+++ b/docs/user_docs/backup-and-restore/backup/backup-repo.md
@@ -130,7 +130,7 @@ If you do not configure the BackupRepo information when installing KubeBlocks, y
--provider oss \
--region cn-zhangjiakou \
--bucket test-kb-backup \
- # --endpoint oss-cn-zhangjiakou-internal.aliyuncs.com \ To display the specified oss endpoint
+ # --endpoint https://oss-cn-zhangjiakou-internal.aliyuncs.com \ To display the specified oss endpoint
--access-key-id \
--secret-access-key \
--default
diff --git a/docs/user_docs/cli/kbcli_backup_create.md b/docs/user_docs/cli/kbcli_backup_create.md
index dd77401b065..5dbced23bc3 100644
--- a/docs/user_docs/cli/kbcli_backup_create.md
+++ b/docs/user_docs/cli/kbcli_backup_create.md
@@ -26,8 +26,8 @@ kbcli backup create NAME [flags]
```
--cluster string Cluster name
-h, --help help for create
+ --method string Backup type (default "snapshot")
--policy string Backup policy name, this flag will be ignored when backup-type is snapshot
- --type string Backup type (default "snapshot")
```
### Options inherited from parent commands
diff --git a/docs/user_docs/cli/kbcli_cluster_backup.md b/docs/user_docs/cli/kbcli_cluster_backup.md
index 483f8772985..28b11ab010a 100644
--- a/docs/user_docs/cli/kbcli_cluster_backup.md
+++ b/docs/user_docs/cli/kbcli_cluster_backup.md
@@ -30,9 +30,9 @@ kbcli cluster backup NAME [flags]
```
-h, --help help for backup
+ --method string Backup method that defined in backup policy
--name string Backup name
--policy string Backup policy name, this flag will be ignored when backup-type is snapshot
- --type string Backup type (default "snapshot")
```
### Options inherited from parent commands
diff --git a/docs/user_docs/cli/kbcli_cluster_create.md b/docs/user_docs/cli/kbcli_cluster_create.md
index f17a5bde74d..658d863ca05 100644
--- a/docs/user_docs/cli/kbcli_cluster_create.md
+++ b/docs/user_docs/cli/kbcli_cluster_create.md
@@ -112,11 +112,11 @@ kbcli cluster create [NAME] [flags]
--restore-to-time string Set a time for point in time recovery
--set stringArray Set the cluster resource including cpu, memory, replicas and storage, each set corresponds to a component.(e.g. --set cpu=1,memory=1Gi,replicas=3,storage=20Gi or --set class=general-1c1g)
-f, --set-file string Use yaml file, URL, or stdin to set the cluster resource
- --source-cluster string Set a source cluster for point in time recovery
--tenancy string Tenancy options, one of: (SharedNode, DedicatedNode) (default "SharedNode")
--termination-policy string Termination policy, one of: (DoNotTerminate, Halt, Delete, WipeOut) (default "Delete")
--tolerations strings Tolerations for cluster, such as "key=value:effect, key:effect", for example '"engineType=mongo:NoSchedule", "diskType:NoSchedule"'
--topology-keys stringArray Topology keys for affinity
+ --volume-restore-policy string the volume claim restore policy, supported values: [Serial, Parallel] (default "Parallel")
```
### Options inherited from parent commands
diff --git a/docs/user_docs/cli/kbcli_cluster_restore.md b/docs/user_docs/cli/kbcli_cluster_restore.md
index afe98b250e8..bb69305794e 100644
--- a/docs/user_docs/cli/kbcli_cluster_restore.md
+++ b/docs/user_docs/cli/kbcli_cluster_restore.md
@@ -22,10 +22,10 @@ kbcli cluster restore [flags]
### Options
```
- --backup string Backup name
- -h, --help help for restore
- --restore-to-time string point in time recovery(PITR)
- --source-cluster string source cluster name
+ --backup string Backup name
+ -h, --help help for restore
+ --restore-to-time string point in time recovery(PITR)
+ --volume-restore-policy string the volume claim restore policy, supported values: [Serial, Parallel] (default "Parallel")
```
### Options inherited from parent commands
diff --git a/docs/user_docs/kubeblocks-for-kafka/configuration/configuration.md b/docs/user_docs/kubeblocks-for-kafka/configuration/configuration.md
index f81f561538c..848d04defa3 100644
--- a/docs/user_docs/kubeblocks-for-kafka/configuration/configuration.md
+++ b/docs/user_docs/kubeblocks-for-kafka/configuration/configuration.md
@@ -7,7 +7,9 @@ sidebar_position: 1
# Configure cluster parameters
-The KubeBlocks configuration function provides a set of consistent default configuration generation strategies for all the databases running on KubeBlocks and also provides a unified parameter configuration interface to facilitate managing parameter reconfiguration, searching the parameter user guide, and validating parameter effectiveness.
+The KubeBlocks configuration function provides a set of consistent default configuration generation strategies for all the databases running on KubeBlocks and also provides a unified parameter configuration interface to facilitate managing parameter configuration, searching the parameter user guide, and validating parameter effectiveness.
+
+From v0.6.0, KubeBlocks supports `kbcli cluster configure` and `kbcli cluster edit-config` to configure parameters. The difference is that KubeBlocks configures parameters automatically with `kbcli cluster configure` but `kbcli cluster edit-config` provides a visualized way for you to edit parameters directly.
## View parameter information
@@ -59,12 +61,12 @@ You can also view the details of this configuration file and parameters.
* Allowed Values: It defines the valid value range of this parameter.
- * Dynamic: The value of `Dynamic` in `Configure Constraint` defines how the parameter reconfiguration takes effect. Currerntly, Kafka only supports static strategy, i.e. `Dynamic` is `false`. Restarting is required to make reconfiguration effective since using kbcli to configure parameters triggers broker restarting.
+ * Dynamic: The value of `Dynamic` in `Configure Constraint` defines how the parameter configuration takes effect. Currerntly, Kafka only supports static strategy, i.e. `Dynamic` is `false`. Restarting is required to make configuration effective.
* Description: It describes the parameter definition.
-## Reconfigure static parameters
+## Configure parameters
-Static parameter reconfiguring requires restarting the pod.
+### Configure parameters with configure command
1. View the current value of `log.cleanup.policy`.
@@ -82,15 +84,15 @@ Static parameter reconfiguring requires restarting the pod.
:::note
- Make sure the value you set is within the Allowed Values of this parameter. Otherwise, the reconfiguration may fail.
+ Make sure the value you set is within the Allowed Values of this parameter. Otherwise, the configuration may fail.
:::
-3. View the status of the parameter reconfiguration.
+3. View the status of the parameter configuration.
- `Status.Progress` and `Status.Status` shows the overall status of the parameter reconfiguration and Conditions show the details.
+ `Status.Progress` and `Status.Status` shows the overall status of the parameter configuration and Conditions show the details.
- When the `Status.Status` shows `Succeed`, the reconfiguration is completed.
+ When the `Status.Status` shows `Succeed`, the configuration is completed.
@@ -135,7 +137,7 @@ Static parameter reconfiguring requires restarting the pod.
-4. View the configuration file to verify whether the parameter is modified.
+4. View the configuration file to verify whether the parameter is configured as expected.
The whole searching process has a 30-second delay.
@@ -146,11 +148,48 @@ Static parameter reconfiguring requires restarting the pod.
mykafka-reconfiguring-wvqns mykafka broker kafka-configuration-tpl server.properties Succeed restart 1/1 Sep 14,2023 16:28 UTC+0800 {"server.properties":"{\"log.cleanup.policy\":\"compact\"}"}
```
+### Configure parameters with edit-config command
+
+For your convenience, KubeBlocks offers a tool `edit-config` to help you to configure parameter in a visulized way.
+
+For Linux and macOS, you can edit configuration files by vi. For Windows, you can edit files on notepad.
+
+1. Edit the configuration file.
+
+ ```bash
+ kbcli cluster edit-config mykafka
+ ```
+
+:::note
+
+If there are multiple components in a cluster, use `--component` to specify a component.
+
+:::
+
+2. View the status of the parameter configuration.
+
+ ```bash
+ kbcli cluster describe-ops xxx -n default
+ ```
+
+3. Connect to the database to verify whether the parameters are configured as expected.
+
+ ```bash
+ kbcli cluster connect mykafka
+ ```
+
+:::note
+
+1. For the `edit-config` function, static parameters and dynamic parameters cannot be edited at the same time.
+2. Deleting a parameter will be supported later.
+
+:::
+
## View history and compare differences
-After the reconfiguration is completed, you can search the reconfiguration history and compare the parameter differences.
+After the configuration is completed, you can search the configuration history and compare the parameter differences.
-View the parameter reconfiguration history.
+View the parameter configuration history.
```bash
kbcli cluster describe-config mykafka
diff --git a/docs/user_docs/kubeblocks-for-mongodb/configuration/configuration.md b/docs/user_docs/kubeblocks-for-mongodb/configuration/configuration.md
index bd2d8fc7ef6..d87950cc4bc 100644
--- a/docs/user_docs/kubeblocks-for-mongodb/configuration/configuration.md
+++ b/docs/user_docs/kubeblocks-for-mongodb/configuration/configuration.md
@@ -7,26 +7,28 @@ sidebar_position: 1
# Configure cluster parameters
-The KubeBlocks configuration function provides a set of consistent default configuration generation strategies for all the databases running on KubeBlocks and also provides a unified parameter configuration interface to facilitate managing parameter reconfiguration, searching the parameter user guide, and validating parameter effectiveness.
+The KubeBlocks configuration function provides a set of consistent default configuration generation strategies for all the databases running on KubeBlocks and also provides a unified parameter configuration interface to facilitate managing parameter configuration, searching the parameter user guide, and validating parameter effectiveness.
-## View parameter information
+From v0.6.0, KubeBlocks supports `kbcli cluster configure` and `kbcli cluster edit-config` to configure parameters. The difference is that KubeBlocks configures parameters automatically with `kbcli cluster configure` but `kbcli cluster edit-config` provides a visualized way for you to edit parameters directly.
-View the current configuration file of a cluster.
+## View parameter information
-```bash
-kbcli cluster describe-config mongodb-cluster
->
-ConfigSpecs Meta:
-CONFIG-SPEC-NAME FILE ENABLED TEMPLATE CONSTRAINT RENDERED COMPONENT CLUSTER
-mongodb-config keyfile false mongodb5.0-config-template mongodb-config-constraints mongodb-cluster-replicaset-mongodb-config replicaset mongodb-cluster
-mongodb-config mongodb.conf true mongodb5.0-config-template mongodb-config-constraints mongodb-cluster-replicaset-mongodb-config replicaset mongodb-cluster
-mongodb-metrics-config metrics-config.yaml false mongodb-metrics-config mongodb-cluster-replicaset-mongodb-metrics-config replicaset mongodb-cluster
+* View the current configuration file of a cluster.
-History modifications:
-OPS-NAME CLUSTER COMPONENT CONFIG-SPEC-NAME FILE STATUS POLICY PROGRESS CREATED-TIME VALID-UPDATED
-```
+ ```bash
+ kbcli cluster describe-config mongodb-cluster
+ >
+ ConfigSpecs Meta:
+ CONFIG-SPEC-NAME FILE ENABLED TEMPLATE CONSTRAINT RENDERED COMPONENT CLUSTER
+ mongodb-config keyfile false mongodb5.0-config-template mongodb-config-constraints mongodb-cluster-replicaset-mongodb-config replicaset mongodb-cluster
+ mongodb-config mongodb.conf true mongodb5.0-config-template mongodb-config-constraints mongodb-cluster-replicaset-mongodb-config replicaset mongodb-cluster
+ mongodb-metrics-config metrics-config.yaml false mongodb-metrics-config mongodb-cluster-replicaset-mongodb-metrics-config replicaset mongodb-cluster
+
+ History modifications:
+ OPS-NAME CLUSTER COMPONENT CONFIG-SPEC-NAME FILE STATUS POLICY PROGRESS CREATED-TIME VALID-UPDATED
+ ```
-From the meta information, the cluster `mongodb-cluster` has a configuration file named `mongodb.conf`.
+ From the meta information, the cluster `mongodb-cluster` has a configuration file named `mongodb.conf`.
You can also view the details of this configuration file and parameters.
@@ -36,9 +38,11 @@ You can also view the details of this configuration file and parameters.
kbcli cluster describe-config mongodb-cluster --show-detail
```
-## Reconfigure parameters
+## Configure parameters
+
+### Configure parameters with configure command
-The example below reconfigures velocity to 1.
+The example below configures velocity to 1.
1. Adjust the values of `velocity` to 1.
@@ -70,9 +74,46 @@ The example below reconfigures velocity to 1.
mongodb-cluster-reconfiguring-q8ndn mongodb-cluster mongodb mongodb-config mongodb.conf Succeed restart 3/3 Apr 21,2023 18:56 UTC+0800 {"mongodb.conf":"{\"systemLog\":{\"verbosity\":\"1\"}}"}```
```
-3. Verify change result.
+3. Verify configuration result.
```bash
root@mongodb-cluster-mongodb-0:/# cat etc/mongodb/mongodb.conf |grep verbosity
verbosity: "1"
```
+
+### Configure parameters with edit-config command
+
+For your convenience, KubeBlocks offers a tool `edit-config` to help you to configure parameter in a visulized way.
+
+For Linux and macOS, you can edit configuration files by vi. For Windows, you can edit files on notepad.
+
+1. Edit the configuration file.
+
+ ```bash
+ kbcli cluster edit-config mongodb-cluster
+ ```
+
+:::note
+
+If there are multiple components in a cluster, use `--component` to specify a component.
+
+:::
+
+2. View the status of the parameter configuration.
+
+ ```bash
+ kbcli cluster describe-ops xxx -n default
+ ```
+
+3. Connect to the database to verify whether the parameters are configured as expected.
+
+ ```bash
+ kbcli cluster connect mongodb-cluster
+ ```
+
+:::note
+
+1. For the `edit-config` function, static parameters and dynamic parameters cannot be edited at the same time.
+2. Deleting a parameter will be supported later.
+
+:::
diff --git a/docs/user_docs/kubeblocks-for-mongodb/migration/feature-and-limit-list-mongodb.md b/docs/user_docs/kubeblocks-for-mongodb/migration/feature-and-limit-list-mongodb.md
index 5ed72224880..8bfc5ae2405 100644
--- a/docs/user_docs/kubeblocks-for-mongodb/migration/feature-and-limit-list-mongodb.md
+++ b/docs/user_docs/kubeblocks-for-mongodb/migration/feature-and-limit-list-mongodb.md
@@ -1,6 +1,6 @@
---
title: Full feature and limit list
-description: The full feature and limit list of KubeBlocks migration function for MOngoDB
+description: The full feature and limit list of KubeBlocks migration function for MongoDB
keywords: [mongodb, migration, migrate data in MongoDB to KubeBlocks, full feature, limit]
sidebar_position: 1
sidebar_label: Full feature and limit list
diff --git a/docs/user_docs/kubeblocks-for-mysql/apecloud-mysql-intro/apecloud-mysql-intro.md b/docs/user_docs/kubeblocks-for-mysql/apecloud-mysql-intro/apecloud-mysql-intro.md
index 5b96addf49a..fe1e57c2774 100644
--- a/docs/user_docs/kubeblocks-for-mysql/apecloud-mysql-intro/apecloud-mysql-intro.md
+++ b/docs/user_docs/kubeblocks-for-mysql/apecloud-mysql-intro/apecloud-mysql-intro.md
@@ -24,7 +24,7 @@ ApeCloud MySQL supports four roles, **Leader**, **Follower**, **Candidate**, and
Role | Leader |Follower | Learner | Candidate |
---- |----| ----|----|----|
- **Capcity**|RW/HA|RO/HA|RO|-|
+ **Capability**|RW/HA|RO/HA|RO|-|
![Role_changing](./../../../img/intro_role_changing.png)
diff --git a/docs/user_docs/kubeblocks-for-mysql/configuration/configuration.md b/docs/user_docs/kubeblocks-for-mysql/configuration/configuration.md
index f43767d0259..63f0f72b7e2 100644
--- a/docs/user_docs/kubeblocks-for-mysql/configuration/configuration.md
+++ b/docs/user_docs/kubeblocks-for-mysql/configuration/configuration.md
@@ -7,7 +7,9 @@ sidebar_position: 1
# Configure cluster parameters
-The KubeBlocks configuration function provides a set of consistent default configuration generation strategies for all the databases running on KubeBlocks and also provides a unified parameter configuration interface to facilitate managing parameter reconfiguration, searching the parameter user guide, and validating parameter effectiveness.
+The KubeBlocks configuration function provides a set of consistent default configuration generation strategies for all the databases running on KubeBlocks and also provides a unified parameter configuration interface to facilitate managing parameter configuration, searching the parameter user guide, and validating parameter effectiveness.
+
+From v0.6.0, KubeBlocks supports both `kbcli cluster configure` and `kbcli cluster edit-config` to configure parameters. The difference is that KubeBlocks configures parameters automatically with `kbcli cluster configure` but `kbcli cluster edit-config` provides a visualized way for you to edit parameters directly.
## View parameter information
@@ -30,7 +32,7 @@ You can also view the details of this configuration file and parameters.
* View the parameter description.
```bash
- kbcli cluster explain-config mysql-cluster |head -n 20
+ kbcli cluster explain-config mysql-cluster | head -n 20
```
* View the user guide of a specified parameter.
@@ -59,14 +61,16 @@ You can also view the details of this configuration file and parameters.
* Allowed Values: It defines the valid value range of this parameter.
- * Dynamic: The value of `Dynamic` in `Configure Constraint` defines how the parameter reconfiguration takes effect. There are two different reconfiguration strategies based on the effectiveness type of modified parameters, i.e. **dynamic** and **static**.
- * When `Dynamic` is `true`, it means the effectiveness type of parameters is **dynamic** and can be updated online. Follow the instructions in [Reconfigure dynamic parameters](#reconfigure-dynamic-parameters).
- * When `Dynamic` is `false`, it means the effectiveness type of parameters is **static** and a pod restarting is required to make reconfiguration effective. Follow the instructions in [Reconfigure static parameters](#reconfigure-static-parameters).
+ * Dynamic: The value of `Dynamic` in `Configure Constraint` defines how the parameter configuration takes effect. There are two different configuration strategies based on the effectiveness type of modified parameters, i.e. **dynamic** and **static**.
+ * When `Dynamic` is `true`, it means the effectiveness type of parameters is **dynamic** and can be configured online. Follow the instructions in [Configure dynamic parameters](#configure-dynamic-parameters).
+ * When `Dynamic` is `false`, it means the effectiveness type of parameters is **static** and a pod restarting is required to make configuration effective. Follow the instructions in [Configure static parameters](#configure-static-parameters).
* Description: It describes the parameter definition.
-## Reconfigure dynamic parameters
+## Configure parameters
+
+### Configure parameters with configure command
-The example below reconfigures `max_connection` and `innodb_buffer_pool_size`.
+The example below takes configuring `max_connection` and `innodb_buffer_pool_size` as an example.
1. View the current values of `max_connection` and `innodb_buffer_pool_size`.
@@ -116,9 +120,9 @@ The example below reconfigures `max_connection` and `innodb_buffer_pool_size`.
:::
-3. Search the status of the parameter reconfiguration.
+3. Search the status of the parameter configuration.
- `Status.Progress` shows the overall status of the parameter reconfiguration and `Conditions` show the details.
+ `Status.Progress` shows the overall status of the parameter configuration and `Conditions` show the details.
```bash
kbcli cluster describe-ops mysql-cluster-reconfiguring-z2wvn -n default
@@ -154,7 +158,7 @@ The example below reconfigures `max_connection` and `innodb_buffer_pool_size`.
-4. Connect to the database to verify whether the parameters are modified.
+4. Connect to the database to verify whether the parameters are configured as expected.
The whole searching process has a 30-second delay since it takes some time for kubelet to synchronize modifications to the volume of the pod.
@@ -184,112 +188,51 @@ The example below reconfigures `max_connection` and `innodb_buffer_pool_size`.
1 row in set (0.00 sec)
```
-## Reconfigure static parameters
+### Configure parameters with edit-config command
-Static parameter reconfiguring requires restarting the pod. The following example reconfigures `ngram_token_size`.
+For your convenience, KubeBlocks offers a tool `edit-config` to help you to configure parameter in a visulized way.
-1. Search the current value of `ngram_token_size` and the default value is 2.
-
- ```bash
- kbcli cluster connect mysql-cluster
- ```
+For Linux and macOS, you can edit configuration files by vi. For Windows, you can edit files on notepad.
- ```bash
- mysql> show variables like '%ngram_token_size%';
- >
- +------------------+-------+
- | Variable_name | Value |
- +------------------+-------+
- | ngram_token_size | 2 |
- +------------------+-------+
- 1 row in set (0.01 sec)
- ```
+The following steps take configuring MySQL Standalone as an example.
-2. Adjust the value of `ngram_token_size`.
+1. Edit the configuration file.
```bash
- kbcli cluster configure mysql-cluster --set=ngram_token_size=6
+ kbcli cluster edit-config mysql-cluster --config-spec=mysql-consensusset-config
```
- :::note
-
- Make sure the value you set is within the Allowed Values of this parameter. Otherwise, the reconfiguration may fail.
+:::note
- :::
+* ApeCloud MySQL currently supports multiple configuration templates, so `--config-spec` is required.
+* If there are multiple components in a cluster, use `--component` to specify a component.
-3. View the status of the parameter reconfiguration.
+:::
- `Status.Progress` and `Status.Status` shows the overall status of the parameter reconfiguration and Conditions show the details.
-
- When the `Status.Status` shows `Succeed`, the reconfiguration is completed.
-
-
-
- Output
+2. View the status of the parameter configuration.
```bash
- # In progress
- kbcli cluster describe-ops mysql-cluster-reconfiguring-nrnpf -n default
- >
- Spec:
- Name: mysql-cluster-reconfiguring-nrnpf NameSpace: default Cluster: mysql-cluster Type: Reconfiguring
-
- Command:
- kbcli cluster configure mysql-cluster --component-names=mysql --template-name=mysql-consensusset-config --config-file=my.cnf --set ngram_token_size=6
-
- Status:
- Start Time: Mar 13,2023 03:37 UTC+0800
- Duration: 22s
- Status: Running
- Progress: 0/1
- OBJECT-KEY STATUS DURATION MESSAGE
- ```
-
- ```bash
- # Parameter reconfiguration is completed
- kbcli cluster describe-ops mysql-cluster-reconfiguring-nrnpf -n default
- >
- Spec:
- Name: mysql-cluster-reconfiguring-nrnpf NameSpace: default Cluster: mysql-cluster Type: Reconfiguring
-
- Command:
- kbcli cluster configure mysql-cluster --component-names=mysql --template-name=mysql-consensusset-config --config-file=my.cnf --set ngram_token_size=6
-
- Status:
- Start Time: Mar 13,2023 03:37 UTC+0800
- Completion Time: Mar 13,2023 03:37 UTC+0800
- Duration: 26s
- Status: Succeed
- Progress: 1/1
- OBJECT-KEY STATUS DURATION MESSAGE
+ kbcli cluster describe-ops xxx -n default
```
-
-
-4. Connect to the database to verify whether the parameters are modified.
-
- The whole searching process has a 30-second delay since it takes some time for kubelete to synchronize modifications to the volume of the pod.
+3. Connect to the database to verify whether the parameters are configured as expected.
```bash
kbcli cluster connect mysql-cluster
```
- ```bash
- mysql> show variables like '%ngram_token_size%';
- >
- +------------------+-------+
- | Variable_name | Value |
- +------------------+-------+
- | ngram_token_size | 6 |
- +------------------+-------+
- 1 row in set (0.09 sec)
- ```
+:::note
+
+1. For the `edit-config` function, static parameters and dynamic parameters cannot be edited at the same time.
+2. Deleting a parameter will be supported later.
+
+:::
## View history and compare differences
-After the reconfiguration is completed, you can search the reconfiguration history and compare the parameter differences.
+After the configuration is completed, you can search the configuration history and compare the parameter differences.
-View the parameter reconfiguration history.
+View the parameter configuration history.
```bash
kbcli cluster describe-config mysql-cluster
diff --git a/docs/user_docs/kubeblocks-for-postgresql/cluster-management/restart-a-postgresql-cluster.md b/docs/user_docs/kubeblocks-for-postgresql/cluster-management/restart-a-postgresql-cluster.md
index 6ac3ff8d308..904bc06ba93 100644
--- a/docs/user_docs/kubeblocks-for-postgresql/cluster-management/restart-a-postgresql-cluster.md
+++ b/docs/user_docs/kubeblocks-for-postgresql/cluster-management/restart-a-postgresql-cluster.md
@@ -28,7 +28,7 @@ Restarting a PostgreSQL cluster triggers a concurrent restart and the leader may
Configure the values of `components` and `ttlSecondsAfterSucceed` and run the command below to restart a specified cluster.
```bash
- kbcli cluster restart NAME --components="pg-replication" \
+ kbcli cluster restart NAME --components="postgresql" \
--ttlSecondsAfterSucceed=30
```
@@ -49,7 +49,7 @@ Restarting a PostgreSQL cluster triggers a concurrent restart and the leader may
clusterRef: pg-cluster
type: Restart
restart:
- - componentName: pg-replication
+ - componentName: postgresql
EOF
```
diff --git a/docs/user_docs/kubeblocks-for-postgresql/configuration/configuration.md b/docs/user_docs/kubeblocks-for-postgresql/configuration/configuration.md
index f02b4c8cd44..80d43cb2362 100644
--- a/docs/user_docs/kubeblocks-for-postgresql/configuration/configuration.md
+++ b/docs/user_docs/kubeblocks-for-postgresql/configuration/configuration.md
@@ -7,7 +7,9 @@ sidebar_position: 1
# Configure cluster parameters
-The KubeBlocks configuration function provides a set of consistent default configuration generation strategies for all the databases running on KubeBlocks and also provides a unified parameter configuration interface to facilitate managing parameter reconfiguration, searching the parameter user guide, and validating parameter effectiveness.
+The KubeBlocks configuration function provides a set of consistent default configuration generation strategies for all the databases running on KubeBlocks and also provides a unified interface to facilitate managing parameter configuration, searching the parameter user guide, and validating parameter effectiveness.
+
+From v0.6.0, KubeBlocks supports `kbcli cluster configure` and `kbcli cluster edit-config` to configure parameters. The difference is that KubeBlocks configures parameters automatically with `kbcli cluster configure` but `kbcli cluster edit-config` provides a visualized way for you to edit parameters directly.
## View parameter information
@@ -30,7 +32,7 @@ You can also view the details of this configuration file and parameters.
* View the parameter description.
```bash
- kbcli cluster explain-config pg-cluster |head -n 20
+ kbcli cluster explain-config pg-cluster | head -n 20
```
* View the user guide of a specified parameter.
@@ -58,14 +60,16 @@ You can also view the details of this configuration file and parameters.
* Allowed Values: It defines the valid value range of this parameter.
- * Dynamic: The value of `Dynamic` in `Configure Constraint` defines how the parameter reconfiguration takes effect. There are two different reconfiguration strategies based on the effectiveness type of modified parameters, i.e. **dynamic** and **static**.
- * When `Dynamic` is `true`, it means the effectiveness type of parameters is **dynamic** and can be updated online. Follow the instructions in [Reconfigure dynamic parameters](#reconfigure-dynamic-parameters).
- * When `Dynamic` is `false`, it means the effectiveness type of parameters is **static** and a pod restarting is required to make reconfiguration effective. Follow the instructions in [Reconfigure static parameters](#reconfigure-static-parameters).
+ * Dynamic: The value of `Dynamic` in `Configure Constraint` defines how the parameter configuration takes effect. There are two different configuration strategies based on the effectiveness type of modified parameters, i.e. **dynamic** and **static**.
+ * When `Dynamic` is `true`, it means the effectiveness type of parameters is **dynamic** and can be configured online. Follow the instructions in [Configure dynamic parameters](#configure-dynamic-parameters).
+ * When `Dynamic` is `false`, it means the effectiveness type of parameters is **static** and a pod restarting is required to make configuration effective. Follow the instructions in [Configure static parameters](#configure-static-parameters).
* Description: It describes the parameter definition.
-## Reconfigure dynamic parameters
+## Configure parameters
+
+### Configure parameters with configure command
-The example below reconfigures `max_connections`.
+The example below takes configuring `max_connections` as an example.
1. View the current values of `max_connections`.
@@ -101,11 +105,11 @@ The example below reconfigures `max_connections`.
:::
-3. View the status of the parameter reconfiguration.
+3. View the status of the parameter configuration.
- `Status.Progress` and `Status.Status` shows the overall status of the parameter reconfiguration and `Conditions` show the details.
+ `Status.Progress` and `Status.Status` shows the overall status of the parameter configuration and `Conditions` show the details.
- When the `Status.Status` shows `Succeed`, the reconfiguration is completed.
+ When the `Status.Status` shows `Succeed`, the configuration is completed.
```bash
kbcli cluster describe-ops pg-cluster-reconfiguring-fq6q7 -n default
@@ -142,7 +146,7 @@ The example below reconfigures `max_connections`.
-4. Connect to the database to verify whether the parameters are modified.
+4. Connect to the database to verify whether the parameter is configured as expected.
The whole searching process has a 30-second delay since it takes some time for kubelet to synchronize modifications to the volume of the pod.
@@ -158,104 +162,48 @@ The example below reconfigures `max_connections`.
(1 row)
```
-## Reconfigure static parameters
+### Configure parameters with edit-config command
-The example below reconfigures `shared_buffers`.
+For your convenience, KubeBlocks offers a tool `edit-config` to help you to configure parameter in a visulized way.
-1. View the current values of `shared_buffers`.
+For Linux and macOS, you can edit configuration files by vi. For Windows, you can edit files on notepad.
- ```bash
- kbcli cluster connect pg-cluster
- ```
+1. Edit the configuration file.
```bash
- postgres=# show shared_buffers;
- shared_buffers
- ----------------
- 256MB
- (1 row)
+ kbcli cluster edit-config pg-cluster
```
-2. Adjust the values of `shared_buffers`.
+:::note
- ```bash
- kbcli cluster configure pg-cluster --set=shared_buffers=512M
- ```
+If there are multiple components in a cluster, use `--component` to specify a component.
- :::note
+:::
- Make sure the value you set is within the Allowed Values of this parameter. If you set a value that does not meet the value range, the system prompts an error. For example,
+2. View the status of the parameter configuration.
```bash
- kbcli cluster configure pg-cluster --set=shared_buffers=5M
- error: failed to validate updated config: [failed to cue template render configure: [pg.maxclients: invalid value 5 (out of bound 16-107374182):
- 343:34
- ]
- ]
+ kbcli cluster describe-ops xxx -n default
```
- :::
-
-3. View the status of the parameter reconfiguration.
-
- `Status.Progress` and `Status.Status` shows the overall status of the parameter reconfiguration and `Conditions` show the details.
-
- When the `Status.Status` shows `Succeed`, the reconfiguration is completed.
+3. Connect to the database to verify whether the parameters are configured as expected.
```bash
- kbcli cluster describe-ops pg-cluster-reconfiguring-rcnzb -n default
- ```
-
-
-
- Output
-
- ```bash
- Spec:
- Name: pg-cluster-reconfiguring-rcnzb NameSpace: default Cluster: pg-cluster Type: Reconfiguring
-
- Command:
- kbcli cluster configure pg-cluster --components=postgresql --config-spec=postgresql-configuration --config-file=postgresql.conf --set shared_buffers=512M --namespace=default
-
- Status:
- Start Time: Mar 17,2023 19:31 UTC+0800
- Duration: 2s
- Status: Running
- Progress: 0/1
- OBJECT-KEY STATUS DURATION MESSAGE
-
- Conditions:
- LAST-TRANSITION-TIME TYPE REASON STATUS MESSAGE
- Mar 17,2023 19:31 UTC+0800 Progressing OpsRequestProgressingStarted True Start to process the OpsRequest: pg-cluster-reconfiguring-rcnzb in Cluster: pg-cluster
- Mar 17,2023 19:31 UTC+0800 Validated ValidateOpsRequestPassed True OpsRequest: pg-cluster-reconfiguring-rcnzb is validated
- Mar 17,2023 19:31 UTC+0800 Reconfigure ReconfigureStarted True Start to reconfigure in Cluster: pg-cluster, Component: postgresql
- Mar 17,2023 19:31 UTC+0800 ReconfigureMerged ReconfigureMerged True Reconfiguring in Cluster: pg-cluster, Component: postgresql, ConfigSpec: postgresql-configuration, info: updated: map[postgresql.conf:{"shared_buffers":"512M"}], added: map[], deleted:map[]
- Mar 17,2023 19:31 UTC+0800 ReconfigureRunning ReconfigureRunning True Reconfiguring in Cluster: pg-cluster, Component: postgresql, ConfigSpec: postgresql-configuration
+ kbcli cluster connect pg-cluster
```
-
-
-4. Connect to the database to verify whether the parameters are modified.
+:::note
- The whole searching process has a 30-second delay since it takes some time for kubelete to synchronize modifications to the volume of the pod.
-
- ```bash
- kbcli cluster connect pg-cluster
- ```
+1. For the `edit-config` function, static parameters and dynamic parameters cannot be edited at the same time.
+2. Deleting a parameter will be supported later.
- ```bash
- postgres=# show shared_buffers;
- shared_buffers
- ----------------
- 512MB
- (1 row)
- ```
+:::
## View history and compare differences
-After the reconfiguration is completed, you can search the reconfiguration history and compare the parameter differences.
+After the configuration is completed, you can search the configuration history and compare the parameter differences.
-View the parameter reconfiguration history.
+View the parameter configuration history.
```bash
kbcli cluster describe-config pg-cluster
diff --git a/docs/user_docs/kubeblocks-for-pulsar/cluster-management/create-pulsar-cluster-on-kubeblocks.md b/docs/user_docs/kubeblocks-for-pulsar/cluster-management/create-pulsar-cluster-on-kubeblocks.md
index a37e2a10b36..d0a04d1116e 100644
--- a/docs/user_docs/kubeblocks-for-pulsar/cluster-management/create-pulsar-cluster-on-kubeblocks.md
+++ b/docs/user_docs/kubeblocks-for-pulsar/cluster-management/create-pulsar-cluster-on-kubeblocks.md
@@ -15,13 +15,14 @@ KubeBlocks supports Pulsar's daily operations, including basic lifecycle operati
## Environment Recommendation
Refer to the Pulsar official document for the configuration, such as memory, cup, and storage, of each component.
+
| Components | Replicas |
| :-------------------- | :------------------------------------------------------------------------ |
| zookeeper | 1 for test environment or 3 for production environment |
| bookies | at lease 3 for test environment, at lease 4 for production environment |
| broker | at least 1, for production environment, 3 replicas recommended |
-| recovery(Optional) | 1; if autoRecovery is not enabled for bookie, at least 3 replicas needed |
-| proxy(Optional) | 1; and for production environment, 3 replicas needed |
+| recovery (Optional) | 1; if autoRecovery is not enabled for bookie, at least 3 replicas needed |
+| proxy (Optional) | 1; and for production environment, 3 replicas needed |
## Create Pulsar cluster
diff --git a/docs/user_docs/kubeblocks-for-pulsar/configuration/configuration.md b/docs/user_docs/kubeblocks-for-pulsar/configuration/configuration.md
index 3636237886b..1f0983f4b6b 100644
--- a/docs/user_docs/kubeblocks-for-pulsar/configuration/configuration.md
+++ b/docs/user_docs/kubeblocks-for-pulsar/configuration/configuration.md
@@ -7,6 +7,8 @@ sidebar_position: 4
# Configure cluster parameters
+From v0.6.0, KubeBlocks supports `kbcli cluster configure` and `kbcli cluster edit-config` to configure parameters. The difference is that KubeBlocks configures parameters automatically with `kbcli cluster configure` but `kbcli cluster edit-config` provides a visualized way for you to edit parameters directly.
+
There are 3 types of parameters:
1. Environment parameters, such as GC-related parameters, `PULSAR_MEM`, and `PULSAR_GC`, changes will apply to all components;
@@ -38,12 +40,17 @@ kbcli cluster describe-config pulsar
* View the parameter description.
```bash
- kbcli cluster explain-config pulsar |head -n 20
+ kbcli cluster explain-config pulsar | head -n 20
```
-## Reconfigure environment parameters
+## Configure parameters
+
+### Configure parameters with configure command
+
+#### Configure environment parameters
***Steps***
+
1. You need to specify the component name to configure parameters. Get the pulsar cluster component name.
```bash
@@ -85,9 +92,9 @@ kbcli cluster describe-config pulsar
kubectl get pod -l app.kubernetes.io/name=pulsar
```
-## Reconfigure dynamic parameters
+#### Configure other parameters
-The following steps take the reconfiguration of dynamic parameter `brokerShutdownTimeoutMs` as an example.
+The following steps take the configuration of dynamic parameter `brokerShutdownTimeoutMs` as an example.
***Steps***
@@ -103,7 +110,7 @@ The following steps take the reconfiguration of dynamic parameter `brokerShutdow
broker-config broker.conf true pulsar-broker-config-tpl brokers-config-constraints pulsar-broker-broker-config broker pulsar
```
-2. Reconfigure parameters.
+2. Configure parameters.
```bash
kbcli cluster configure pulsar --component=broker --config-spec=broker-config --set brokerShutdownTimeoutMs=66600
@@ -136,69 +143,46 @@ The following steps take the reconfiguration of dynamic parameter `brokerShutdow
OBJECT-KEY STATUS DURATION MESSAGE
```
-## Reconfigure static parameters
-
-Static parameter reconfiguring requires restarting the pod. The following example reconfigures `lostBookieRecoveryDelay`.
+### Configure parameters with edit-config command
-1. Get the current configuration.
+For your convenience, KubeBlocks offers a tool `edit-config` to help you to configure parameter in a visulized way.
- ```bash
- kbcli cluster explain-config pulsar --component=broker
- >
- ConfigSpecs Meta:
- CONFIG-SPEC-NAME FILE ENABLED TEMPLATE CONSTRAINT RENDERED COMPONENT CLUSTER
- agamotto-configuration agamotto-config.yaml false pulsar-agamotto-conf-tpl pulsar-broker-agamotto-configuration broker pulsar
- broker-env conf true pulsar-broker-env-tpl pulsar-env-constraints pulsar-broker-broker-env broker pulsar
- broker-config broker.conf true pulsar-broker-config-tpl brokers-config-constraints pulsar-broker-broker-config broker pulsar
- ```
+For Linux and macOS, you can edit configuration files by vi. For Windows, you can edit files on notepad.
-2. Adjust the value of `lostBookieRecoveryDelay`.
+1. Edit the configuration file.
```bash
- kbcli cluster configure pulsar --component=broker --config-spec=broker-config --set lostBookieRecoveryDelay=1000
+ kbcli cluster edit-config pulsar
```
- :::note
+:::note
- The change of parameters may cause the restart of the cluster. Enter `yes` to confirm it.
+If there are multiple components in a cluster, use `--component` to specify a component.
- :::
+:::
- ***Example***
+2. View the status of the parameter configuration.
```bash
- kbcli cluster configure pulsar --component=broker --config-spec=broker-config --set lostBookieRecoveryDelay=1000
- >
- Warning: The parameter change incurs a cluster restart, which brings the cluster down for a while. Enter to continue...
- Please type "yes" to confirm: yes
- Will updated configure file meta:
- ConfigSpec: broker-config ConfigFile: broker.conf ComponentName: broker ClusterName: pulsar
- OpsRequest pulsar-reconfiguring-gmz7w created successfully, you can view the progress:
- kbcli cluster describe-ops pulsar-reconfiguring-gmz7w -n default
+ kbcli cluster describe-ops xxx -n default
```
-3. View the status of the parameter reconfiguration.
+3. Connect to the database to verify whether the parameters are configured as expected.
```bash
- kbcli cluster describe-ops pulsar-reconfiguring-gmz7w -n default
- >
- Spec:
- Name: pulsar-reconfiguring-gmz7w NameSpace: default Cluster: pulsar Type: Reconfiguring
+ kbcli cluster connect pulsar
+ ```
- Command:
- kbcli cluster configure pulsar --components=broker --config-spec=broker-config --config-file=broker.conf --set lostBookieRecoveryDelay=1000 --namespace=default
+:::note
- Status:
- Start Time: Jul 20,2023 09:57 UTC+0800
- Duration: 57s
- Status: Running
- Progress: 1/2
- OBJECT-KEY STATUS DURATION MESSAGE
- ```
+1. For the `edit-config` function, static parameters and dynamic parameters cannot be edited at the same time.
+2. Deleting a parameter will be supported later.
+
+:::
-## Reconfigure with kubectl
+### Configure parameters with kubectl
-Using kubectl to reconfigure pulsar cluster requires modifying the configuration file.
+Using kubectl to configure pulsar cluster requires modifying the configuration file.
***Steps***
diff --git a/docs/user_docs/kubeblocks-for-redis/configuration/configuration.md b/docs/user_docs/kubeblocks-for-redis/configuration/configuration.md
index 4b2a0ff4436..8dff5a7010d 100644
--- a/docs/user_docs/kubeblocks-for-redis/configuration/configuration.md
+++ b/docs/user_docs/kubeblocks-for-redis/configuration/configuration.md
@@ -7,7 +7,9 @@ sidebar_position: 1
# Configure cluster parameters
-The KubeBlocks configuration function provides a set of consistent default configuration generation strategies for all the databases running on KubeBlocks and also provides a unified parameter configuration interface to facilitate managing parameter reconfiguration, searching the parameter user guide, and validating parameter effectiveness.
+The KubeBlocks configuration function provides a set of consistent default configuration generation strategies for all the databases running on KubeBlocks and also provides a unified parameter configuration interface to facilitate managing parameter configuration, searching the parameter user guide, and validating parameter effectiveness.
+
+From v0.6.0, KubeBlocks supports `kbcli cluster configure` and `kbcli cluster edit-config` to configure parameters. The difference is that KubeBlocks configures parameters automatically with `kbcli cluster configure` but `kbcli cluster edit-config` provides a visualized way for you to edit parameters directly.
## View parameter information
@@ -58,14 +60,16 @@ You can also view the details of this configuration file and parameters.
* Allowed Values: It defines the valid value range of this parameter.
- * Dynamic: The value of `Dynamic` in `Configure Constraint` defines how the parameter reconfiguration takes effect. There are two different reconfiguration strategies based on the effectiveness type of modified parameters, i.e. **dynamic** and **static**.
- * When `Dynamic` is `true`, it means the effectiveness type of parameters is **dynamic** and can be updated online. Follow the instructions in [Reconfigure dynamic parameters](#reconfigure-dynamic-parameters).
- * When `Dynamic` is `false`, it means the effectiveness type of parameters is **static** and a pod restarting is required to make reconfiguration effective. Follow the instructions in [Reconfigure static parameters](#reconfigure-static-parameters).
+ * Dynamic: The value of `Dynamic` in `Configure Constraint` defines how the parameter configuration takes effect. There are two different configuration strategies based on the effectiveness type of modified parameters, i.e. **dynamic** and **static**.
+ * When `Dynamic` is `true`, it means the effectiveness type of parameters is **dynamic** and can be updated online. Follow the instructions in [Configure dynamic parameters](#configure-dynamic-parameters).
+ * When `Dynamic` is `false`, it means the effectiveness type of parameters is **static** and a pod restarting is required to make configuration effective. Follow the instructions in [Configure static parameters](#configure-static-parameters).
* Description: It describes the parameter definition.
-## Reconfigure dynamic parameters
+## Configure parameters
+
+### Configure parameters with configure command
-The example below reconfigures `acllog-max-len`.
+The example below configures `acllog-max-len`.
1. View the current values of `acllog-max-len`.
@@ -104,11 +108,11 @@ The example below reconfigures `acllog-max-len`.
:::
-3. View the status of the parameter reconfiguration.
+3. View the status of the parameter configuration.
- `Status.Progress` and `Status.Status` shows the overall status of the parameter reconfiguration and `Conditions` show the details.
+ `Status.Progress` and `Status.Status` shows the overall status of the parameter configuration and `Conditions` show the details.
- When the `Status.Status` shows `Succeed`, the reconfiguration is completed.
+ When the `Status.Status` shows `Succeed`, the configuration is completed.
```bash
kbcli cluster describe-ops redis-cluster-reconfiguring-zjztm -n default
@@ -141,7 +145,7 @@ The example below reconfigures `acllog-max-len`.
-4. Connect to the database to verify whether the parameters are modified.
+4. Connect to the database to verify whether the parameter is configured as expected.
The whole searching process has a 30-second delay since it takes some time for kubelet to synchronize modifications to the volume of the pod.
@@ -155,108 +159,48 @@ The example below reconfigures `acllog-max-len`.
2) "256"
```
-## Reconfigure static parameters
+### Configure parameters with edit-config command
-The example below reconfigures `maxclients` and `databases`.
+For your convenience, KubeBlocks offers a tool `edit-config` to help you to configure parameter in a visulized way.
-1. View the current values of `maxclients` and `databases`.
+For Linux and macOS, you can edit configuration files by vi. For Windows, you can edit files on notepad.
- ```bash
- kbcli cluster connect redis-cluster
- ```
+1. Edit the configuration file.
```bash
- 127.0.0.1:6379> config get parameter maxclients databases
- 1) "databases"
- 2) "16"
- 3) "maxclients"
- 4) "10000"
+ kbcli cluster edit-config redis-cluster
```
-2. Adjust the values of `maxclients` and `databases`.
+:::note
- ```bash
- kbcli cluster configure redis-cluster --component=redis --set=maxclients=20000,databases=32
- ```
+If there are multiple components in a cluster, use `--component` to specify a component.
- :::note
+:::
- Make sure the value you set is within the Allowed Values of this parameter. If you set a value that does not meet the value range, the system prompts an error. For example,
+2. View the status of the parameter configuration.
```bash
- kbcli cluster configure redis-cluster --component=redis --set=maxclients=65001
- >
- error: failed to validate updated config: [failed to cue template render configure: [configuration.maxclients: 2 errors in empty disjunction:
- configuration.maxclients: conflicting values 65000 and 65001:
- 100:37
- 155:16
- configuration.maxclients: invalid value 65001 (out of bound <=65000):
- 100:26
- ]
- ]
+ kbcli cluster describe-ops xxx -n default
```
- :::
-
-3. View the status of the parameter reconfiguration.
-
- `Status.Progress` and `Status.Status` shows the overall status of the parameter reconfiguration and `Conditions` show the details.
-
- When the `Status.Status` shows `Succeed`, the reconfiguration is completed.
+3. Connect to the database to verify whether the parameters are configured as expected.
```bash
- kbcli cluster describe-ops redis-cluster-reconfiguring-zrkq7 -n default
- ```
-
-
- Output
-
- ```bash
- Spec:
- Name: redis-cluster-reconfiguring-zrkq7 NameSpace: default Cluster: redis-cluster Type: Reconfiguring
-
- Command:
- kbcli cluster configure redis-cluster --components=redis --config-spec=redis-replication-config --config-file=redis.conf --set databases=32 --set maxclients=20000 --namespace=default
-
- Status:
- Start Time: Apr 17,2023 17:28 UTC+0800
- Duration: 2s
- Status: Running
- Progress: 0/1
- OBJECT-KEY STATUS DURATION MESSAGE
-
- Conditions:
- LAST-TRANSITION-TIME TYPE REASON STATUS MESSAGE
- Apr 17,2023 17:28 UTC+0800 Progressing OpsRequestProgressingStarted True Start to process the OpsRequest: redis-cluster-reconfiguring-zrkq7 in Cluster: redis-cluster
- Apr 17,2023 17:28 UTC+0800 Validated ValidateOpsRequestPassed True OpsRequest: redis-cluster-reconfiguring-zrkq7 is validated
- Apr 17,2023 17:28 UTC+0800 Reconfigure ReconfigureStarted True Start to reconfigure in Cluster: redis-cluster, Component: redis
- Apr 17,2023 17:28 UTC+0800 ReconfigureMerged ReconfigureMerged True Reconfiguring in Cluster: redis-cluster, Component: redis, ConfigSpec: redis-replication-config, info: updated: map[redis.conf:{"databases":"32","maxclients":"20000"}], added: map[], deleted:map[]
- Apr 17,2023 17:28 UTC+0800 ReconfigureRunning ReconfigureRunning True Reconfiguring in Cluster: redis-cluster, Component: redis, ConfigSpec: redis-replication-config
+ kbcli cluster connect redis-cluster
```
-
-
-4. Connect to the database to verify whether the parameters are modified.
-
- The whole searching process has a 30-second delay since it takes some time for kubelete to synchronize modifications to the volume of the pod.
+:::note
- ```bash
- kbcli cluster connect redis-cluster
- ```
+1. For the `edit-config` function, static parameters and dynamic parameters cannot be edited at the same time.
+2. Deleting a parameter will be supported later.
- ```bash
- 127.0.0.1:6379> config get parameter maxclients databases
- 1) "databases"
- 2) "32"
- 3) "maxclients"
- 4) "20000"
- ```
+:::
## View history and compare differences
-After the reconfiguration is completed, you can search the reconfiguration history and compare the parameter differences.
+After the configuration is completed, you can search the configuration history and compare the parameter differences.
-View the parameter reconfiguration history.
+View the parameter configuration history.
```bash
kbcli cluster describe-config redis-cluster --component=redis
diff --git a/docs/user_docs/kubeblocks-for-vector-database/manage-vector-databases.md b/docs/user_docs/kubeblocks-for-vector-database/manage-vector-databases.md
index 0e77cad4c6c..ba577077b31 100644
--- a/docs/user_docs/kubeblocks-for-vector-database/manage-vector-databases.md
+++ b/docs/user_docs/kubeblocks-for-vector-database/manage-vector-databases.md
@@ -39,7 +39,7 @@ Before you start, [Install KubeBlocks](./../installation/install-with-helm/) and
3. Check the cluster information.
```bash
- kblci cluster describe qdrant
+ kbcli cluster describe qdrant
>
Name: qdrant Created Time: Aug 15,2023 23:03 UTC+0800
NAMESPACE CLUSTER-DEFINITION VERSION STATUS TERMINATION-POLICY
@@ -73,11 +73,17 @@ Before you start, [Install KubeBlocks](./../installation/install-with-helm/) and
## Connect to a vector database cluster
-Use the following command to connect to a vector database cluster.
+Qdrant provides both HTTP and gRPC protocols for client access on ports 6333 and 6334 respectively. Depending on where the client is, different connection options are offered to connect to the Qdrant cluster.
-```bash
-kbcli cluster connect --namespace
-```
+:::note
+
+If your cluster is on AWS, install the AWS Load Balancer Controller first.
+
+:::
+
+- If your client is inside a K8s cluster, run `kbcli cluster describe qdrant` to get the ClusterIP address of the cluster or the corresponding K8s cluster domain name.
+- If your client is outside the K8s cluster but in the same VPC as the server, run `kbcli cluster expose qdant --enable=true --type=vpc` to get a VPC load balancer address for the database cluster.
+- If your client is outside the VPC, run `kbcli cluster expose qdant --enable=true --type=internet` to open a public network reachable address for the database cluster.
## Monitor the vector database
diff --git a/go.mod b/go.mod
index 501db1d9405..f72bb23a94e 100644
--- a/go.mod
+++ b/go.mod
@@ -18,27 +18,26 @@ require (
github.com/bhmj/jsonslice v1.1.2
github.com/briandowns/spinner v1.23.0
github.com/cenkalti/backoff/v4 v4.2.1
- github.com/chaos-mesh/chaos-mesh/api v0.0.0-20230423031423-0b31a519b502
+ github.com/chaos-mesh/chaos-mesh/api v0.0.0-20230912020346-a5d89c1c90ad
github.com/clbanning/mxj/v2 v2.5.7
github.com/containerd/stargz-snapshotter/estargz v0.14.3
- github.com/containers/common v0.49.1
+ github.com/containers/common v0.55.4
github.com/dapr/kit v0.11.3
github.com/deckarep/golang-set/v2 v2.3.1
github.com/dlclark/regexp2 v1.10.0
- github.com/docker/docker v24.0.2+incompatible
+ github.com/docker/docker v24.0.6+incompatible
github.com/docker/go-connections v0.4.1-0.20190612165340-fd1b1942c4d5
github.com/dustin/go-humanize v1.0.1
github.com/evanphx/json-patch v5.6.0+incompatible
github.com/fatih/color v1.15.0
github.com/fsnotify/fsnotify v1.6.0
github.com/ghodss/yaml v1.0.0
- github.com/go-errors/errors v1.4.0
+ github.com/go-errors/errors v1.4.2
github.com/go-git/go-git/v5 v5.6.1
github.com/go-logr/logr v1.2.4
github.com/go-logr/zapr v1.2.4
github.com/go-redis/redismock/v9 v9.0.3
github.com/go-sql-driver/mysql v1.7.1
- github.com/go-task/slim-sprig v0.0.0-20230315185526-52ccab3ef572
github.com/golang-jwt/jwt v3.2.2+incompatible
github.com/golang/mock v1.6.0
github.com/google/go-cmp v0.5.9
@@ -50,7 +49,7 @@ require (
github.com/jackc/pgx/v5 v5.4.3
github.com/jedib0t/go-pretty/v6 v6.4.6
github.com/json-iterator/go v1.1.12
- github.com/k3d-io/k3d/v5 v5.4.4
+ github.com/k3d-io/k3d/v5 v5.6.0
github.com/kubernetes-csi/external-snapshotter/client/v3 v3.0.0
github.com/kubernetes-csi/external-snapshotter/client/v6 v6.2.0
github.com/kubesphere/kubekey/v3 v3.0.7
@@ -61,7 +60,7 @@ require (
github.com/mitchellh/mapstructure v1.5.1-0.20220423185008-bf980b35cac4
github.com/onsi/ginkgo/v2 v2.11.0
github.com/onsi/gomega v1.27.8
- github.com/opencontainers/image-spec v1.1.0-rc2
+ github.com/opencontainers/image-spec v1.1.0-rc5
github.com/pashagolub/pgxmock/v2 v2.11.0
github.com/pkg/errors v0.9.1
github.com/pmezard/go-difflib v1.0.0
@@ -87,59 +86,59 @@ require (
github.com/xdg-go/scram v1.1.2
github.com/xeipuuv/gojsonschema v1.2.0
go.etcd.io/etcd/client/v3 v3.5.9
- go.etcd.io/etcd/server/v3 v3.5.6
+ go.etcd.io/etcd/server/v3 v3.5.9
go.mongodb.org/mongo-driver v1.11.6
go.uber.org/automaxprocs v1.5.2
go.uber.org/zap v1.24.0
- golang.org/x/crypto v0.12.0
+ golang.org/x/crypto v0.13.0
golang.org/x/exp v0.0.0-20230711153332-06a737ee72cb
golang.org/x/net v0.14.0
golang.org/x/oauth2 v0.9.0
golang.org/x/sync v0.3.0
- golang.org/x/text v0.12.0
+ golang.org/x/text v0.13.0
google.golang.org/grpc v1.56.2
google.golang.org/protobuf v1.31.0
gopkg.in/inf.v0 v0.9.1
gopkg.in/yaml.v2 v2.4.0
- helm.sh/helm/v3 v3.11.1
- k8s.io/api v0.26.3
- k8s.io/apiextensions-apiserver v0.26.3
- k8s.io/apimachinery v0.26.3
- k8s.io/cli-runtime v0.26.3
- k8s.io/client-go v0.26.3
- k8s.io/code-generator v0.26.3
- k8s.io/component-base v0.26.3
+ helm.sh/helm/v3 v3.12.3
+ k8s.io/api v0.28.2
+ k8s.io/apiextensions-apiserver v0.28.1
+ k8s.io/apimachinery v0.28.2
+ k8s.io/cli-runtime v0.28.2
+ k8s.io/client-go v0.28.2
+ k8s.io/code-generator v0.28.2
+ k8s.io/component-base v0.28.2
k8s.io/cri-api v0.27.1
- k8s.io/gengo v0.0.0-20220913193501-391367153a38
+ k8s.io/gengo v0.0.0-20230829151522-9cce18d56c01
k8s.io/klog v1.0.0
k8s.io/klog/v2 v2.100.1
- k8s.io/kube-openapi v0.0.0-20230308215209-15aac26d736a
- k8s.io/kubectl v0.26.0
+ k8s.io/kube-openapi v0.0.0-20230918164632-68afd615200d
+ k8s.io/kubectl v0.28.2
k8s.io/kubelet v0.26.1
- k8s.io/metrics v0.26.3
- k8s.io/utils v0.0.0-20230406110748-d93618cff8a2
- sigs.k8s.io/controller-runtime v0.14.6
- sigs.k8s.io/kustomize/kyaml v0.13.9
+ k8s.io/metrics v0.28.2
+ k8s.io/utils v0.0.0-20230726121419-3b25d923346b
+ sigs.k8s.io/controller-runtime v0.15.2
+ sigs.k8s.io/kustomize/kyaml v0.14.3
sigs.k8s.io/yaml v1.3.0
)
require (
cloud.google.com/go v0.110.0 // indirect
- cloud.google.com/go/compute v1.19.0 // indirect
+ cloud.google.com/go/compute v1.19.3 // indirect
cloud.google.com/go/compute/metadata v0.2.3 // indirect
cloud.google.com/go/iam v0.13.0 // indirect
cloud.google.com/go/storage v1.29.0 // indirect
+ dario.cat/mergo v1.0.0 // indirect
github.com/99designs/go-keychain v0.0.0-20191008050251-8e49817e8af4 // indirect
- github.com/Azure/azure-sdk-for-go/sdk/azcore v1.1.3 // indirect
- github.com/Azure/azure-sdk-for-go/sdk/internal v1.0.0 // indirect
- github.com/Azure/go-ansiterm v0.0.0-20210617225240-d185dfc1b5a1 // indirect
- github.com/BurntSushi/toml v1.2.1 // indirect
+ github.com/AdaLogics/go-fuzz-headers v0.0.0-20230811130428-ced1acdcaa24 // indirect
+ github.com/Azure/go-ansiterm v0.0.0-20230124172434-306776ec8161 // indirect
+ github.com/BurntSushi/toml v1.3.2 // indirect
github.com/MakeNowJust/heredoc v1.0.0 // indirect
github.com/Masterminds/goutils v1.1.1 // indirect
- github.com/Masterminds/squirrel v1.5.3 // indirect
- github.com/Microsoft/go-winio v0.6.0 // indirect
- github.com/Microsoft/hcsshim v0.9.6 // indirect
- github.com/ProtonMail/go-crypto v0.0.0-20230217124315-7d5c6f04bbb8 // indirect
+ github.com/Masterminds/squirrel v1.5.4 // indirect
+ github.com/Microsoft/go-winio v0.6.1 // indirect
+ github.com/Microsoft/hcsshim v0.11.0 // indirect
+ github.com/ProtonMail/go-crypto v0.0.0-20230528122434-6f98819771a1 // indirect
github.com/VividCortex/ewma v1.2.0 // indirect
github.com/acarl005/stripansi v0.0.0-20180116102854-5a71ef0e047d // indirect
github.com/acomagu/bufpipe v1.0.4 // indirect
@@ -155,18 +154,18 @@ require (
github.com/chzyer/readline v1.5.1 // indirect
github.com/cloudflare/circl v1.3.3 // indirect
github.com/cockroachdb/apd/v3 v3.2.0 // indirect
- github.com/containerd/cgroups v1.0.4 // indirect
- github.com/containerd/containerd v1.6.18 // indirect
- github.com/containers/image/v5 v5.24.0 // indirect
+ github.com/containerd/cgroups v1.1.0 // indirect
+ github.com/containerd/containerd v1.7.6 // indirect
+ github.com/containers/image/v5 v5.26.2 // indirect
github.com/containers/libtrust v0.0.0-20230121012942-c1716e8a8d01 // indirect
github.com/containers/ocicrypt v1.1.7 // indirect
- github.com/containers/storage v1.45.3 // indirect
- github.com/coreos/go-semver v0.3.0 // indirect
- github.com/coreos/go-systemd/v22 v22.3.2 // indirect
+ github.com/containers/storage v1.48.1 // indirect
+ github.com/coreos/go-semver v0.3.1 // indirect
+ github.com/coreos/go-systemd/v22 v22.5.0 // indirect
github.com/cpuguy83/go-md2man/v2 v2.0.2 // indirect
- github.com/cyberphone/json-canonicalization v0.0.0-20220623050100-57a0ce2678a7 // indirect
- github.com/cyphar/filepath-securejoin v0.2.3 // indirect
- github.com/danieljoos/wincred v1.1.2 // indirect
+ github.com/cyberphone/json-canonicalization v0.0.0-20230514072755-504adb8a8af1 // indirect
+ github.com/cyphar/filepath-securejoin v0.2.4 // indirect
+ github.com/danieljoos/wincred v1.2.0 // indirect
github.com/davecgh/go-spew v1.1.1 // indirect
github.com/daviddengcn/go-colortext v1.0.0 // indirect
github.com/deckarep/golang-set v1.8.0 // indirect
@@ -174,9 +173,9 @@ require (
github.com/dgryski/go-rendezvous v0.0.0-20200823014737-9f7001d12a5f // indirect
github.com/dimchansky/utfbom v1.1.1 // indirect
github.com/distribution/distribution/v3 v3.0.0-20221208165359-362910506bc2 // indirect
- github.com/docker/cli v23.0.1+incompatible // indirect
+ github.com/docker/cli v24.0.6+incompatible // indirect
github.com/docker/distribution v2.8.2+incompatible // indirect
- github.com/docker/docker-credential-helpers v0.7.0 // indirect
+ github.com/docker/docker-credential-helpers v0.8.0 // indirect
github.com/docker/go v1.5.1-1.0.20160303222718-d30aec9fd63c // indirect
github.com/docker/go-metrics v0.0.1 // indirect
github.com/docker/go-units v0.5.0 // indirect
@@ -192,24 +191,24 @@ require (
github.com/exponent-io/jsonpath v0.0.0-20151013193312-d6023ce2651d // indirect
github.com/fatih/camelcase v1.0.0 // indirect
github.com/felixge/httpsnoop v1.0.3 // indirect
- github.com/fullstorydev/grpcurl v1.8.7 // indirect
- github.com/fvbommel/sortorder v1.0.2 // indirect
+ github.com/fvbommel/sortorder v1.1.0 // indirect
github.com/go-git/gcfg v1.5.0 // indirect
github.com/go-git/go-billy/v5 v5.4.1 // indirect
- github.com/go-gorp/gorp/v3 v3.0.2 // indirect
+ github.com/go-gorp/gorp/v3 v3.0.5 // indirect
github.com/go-logr/stdr v1.2.2 // indirect
github.com/go-ole/go-ole v1.2.6 // indirect
github.com/go-openapi/analysis v0.21.4 // indirect
github.com/go-openapi/errors v0.20.3 // indirect
github.com/go-openapi/jsonpointer v0.19.6 // indirect
- github.com/go-openapi/jsonreference v0.20.1 // indirect
+ github.com/go-openapi/jsonreference v0.20.2 // indirect
github.com/go-openapi/loads v0.21.2 // indirect
github.com/go-openapi/runtime v0.26.0 // indirect
github.com/go-openapi/spec v0.20.9 // indirect
github.com/go-openapi/strfmt v0.21.7 // indirect
- github.com/go-openapi/swag v0.22.3 // indirect
+ github.com/go-openapi/swag v0.22.4 // indirect
github.com/go-openapi/validate v0.22.1 // indirect
github.com/go-redis/redis/v7 v7.4.1 // indirect
+ github.com/go-task/slim-sprig v0.0.0-20230315185526-52ccab3ef572 // indirect
github.com/go-test/deep v1.1.0 // indirect
github.com/gobwas/glob v0.2.3 // indirect
github.com/godbus/dbus v0.0.0-20190726142602-4481cbc300e2 // indirect
@@ -221,24 +220,24 @@ require (
github.com/golang/snappy v0.0.4 // indirect
github.com/goodhosts/hostsfile v0.1.1 // indirect
github.com/google/btree v1.1.2 // indirect
- github.com/google/gnostic v0.6.9 // indirect
- github.com/google/go-containerregistry v0.14.0 // indirect
+ github.com/google/gnostic-models v0.6.8 // indirect
+ github.com/google/go-containerregistry v0.16.1 // indirect
github.com/google/go-intervals v0.0.2 // indirect
github.com/google/gofuzz v1.2.0 // indirect
- github.com/google/pprof v0.0.0-20221103000818-d260c55eee4c // indirect
- github.com/google/s2a-go v0.1.3 // indirect
+ github.com/google/pprof v0.0.0-20230323073829-e72429f035bd // indirect
+ github.com/google/s2a-go v0.1.4 // indirect
github.com/google/shlex v0.0.0-20191202100458-e7afc7fbc510 // indirect
github.com/googleapis/enterprise-certificate-proxy v0.2.3 // indirect
- github.com/googleapis/gax-go/v2 v2.8.0 // indirect
+ github.com/googleapis/gax-go/v2 v2.9.1 // indirect
github.com/gorilla/handlers v1.5.1 // indirect
github.com/gorilla/mux v1.8.0 // indirect
github.com/gorilla/websocket v1.5.0 // indirect
github.com/gosuri/uitable v0.0.4 // indirect
github.com/gregjones/httpcache v0.0.0-20180305231024-9cad4c3443a7 // indirect
github.com/grpc-ecosystem/go-grpc-middleware v1.4.0 // indirect
- github.com/grpc-ecosystem/go-grpc-prometheus v1.2.0 // indirect
+ github.com/grpc-ecosystem/go-grpc-prometheus v1.2.1-0.20210315223345-82c243799c99 // indirect
github.com/grpc-ecosystem/grpc-gateway v1.16.0 // indirect
- github.com/grpc-ecosystem/grpc-gateway/v2 v2.11.3 // indirect
+ github.com/grpc-ecosystem/grpc-gateway/v2 v2.15.2 // indirect
github.com/gsterjov/go-libsecret v0.0.0-20161001094733-a6f4afe4910c // indirect
github.com/hashicorp/errwrap v1.1.0 // indirect
github.com/hashicorp/go-getter v1.7.0 // indirect
@@ -247,8 +246,8 @@ require (
github.com/hashicorp/go-uuid v1.0.3 // indirect
github.com/hashicorp/hcl v1.0.0 // indirect
github.com/hashicorp/terraform-json v0.15.0 // indirect
- github.com/huandu/xstrings v1.3.3 // indirect
- github.com/imdario/mergo v0.3.13 // indirect
+ github.com/huandu/xstrings v1.4.0 // indirect
+ github.com/imdario/mergo v0.3.14 // indirect
github.com/inconshreveable/mousetrap v1.1.0 // indirect
github.com/jackc/pgpassfile v1.0.0 // indirect
github.com/jackc/pgservicefile v0.0.0-20221227161230-091c0ba34f0a // indirect
@@ -259,14 +258,13 @@ require (
github.com/jcmturner/gofork v1.7.6 // indirect
github.com/jcmturner/gokrb5/v8 v8.4.3 // indirect
github.com/jcmturner/rpc/v2 v2.0.3 // indirect
- github.com/jhump/protoreflect v1.14.1 // indirect
github.com/jmespath/go-jmespath v0.4.0 // indirect
github.com/jmoiron/sqlx v1.3.5 // indirect
github.com/jonboulle/clockwork v0.3.0 // indirect
github.com/josharian/intern v1.0.0 // indirect
github.com/kevinburke/ssh_config v1.2.0 // indirect
- github.com/klauspost/compress v1.16.3 // indirect
- github.com/klauspost/pgzip v1.2.6-0.20220930104621-17e8dac29df8 // indirect
+ github.com/klauspost/compress v1.16.6 // indirect
+ github.com/klauspost/pgzip v1.2.6 // indirect
github.com/kopia/kopia v0.10.7 // indirect
github.com/kr/fs v0.1.0 // indirect
github.com/kubernetes-csi/external-snapshotter/client/v4 v4.2.0 // indirect
@@ -274,7 +272,7 @@ require (
github.com/lann/ps v0.0.0-20150810152359-62de8c46ede0 // indirect
github.com/lestrrat-go/file-rotatelogs v2.4.0+incompatible // indirect
github.com/lestrrat-go/strftime v1.0.5 // indirect
- github.com/letsencrypt/boulder v0.0.0-20221109233200-85aa52084eaf // indirect
+ github.com/letsencrypt/boulder v0.0.0-20230213213521-fdfea0d469b6 // indirect
github.com/lib/pq v1.10.9 // indirect
github.com/liggitt/tabwriter v0.0.0-20181228230101-89fcab3d43de // indirect
github.com/lithammer/dedent v1.1.0 // indirect
@@ -287,17 +285,18 @@ require (
github.com/mattn/go-sqlite3 v2.0.3+incompatible // indirect
github.com/matttproud/golang_protobuf_extensions v1.0.4 // indirect
github.com/miekg/pkcs11 v1.1.1 // indirect
- github.com/mistifyio/go-zfs/v3 v3.0.0 // indirect
+ github.com/mistifyio/go-zfs/v3 v3.0.1 // indirect
github.com/mitchellh/copystructure v1.2.0 // indirect
github.com/mitchellh/go-homedir v1.1.0 // indirect
github.com/mitchellh/go-testing-interface v1.14.1 // indirect
github.com/mitchellh/go-wordwrap v1.0.1 // indirect
github.com/mitchellh/reflectwalk v1.0.2 // indirect
github.com/moby/locker v1.0.1 // indirect
+ github.com/moby/patternmatcher v0.6.0 // indirect
github.com/moby/spdystream v0.2.0 // indirect
- github.com/moby/sys/mount v0.3.0 // indirect
github.com/moby/sys/mountinfo v0.6.2 // indirect
- github.com/moby/term v0.0.0-20221205130635-1aeaba878587 // indirect
+ github.com/moby/sys/sequential v0.5.0 // indirect
+ github.com/moby/term v0.5.0 // indirect
github.com/modern-go/concurrent v0.0.0-20180306012644-bacd9c7ef1dd // indirect
github.com/modern-go/reflect2 v1.0.2 // indirect
github.com/modood/table v0.0.0-20220527013332-8d47e76dad33 // indirect
@@ -311,9 +310,9 @@ require (
github.com/nsf/termbox-go v0.0.0-20190121233118-02980233997d // indirect
github.com/oklog/ulid v1.3.1 // indirect
github.com/opencontainers/go-digest v1.0.0 // indirect
- github.com/opencontainers/runc v1.1.5 // indirect
- github.com/opencontainers/runtime-spec v1.0.3-0.20210326190908-1c3f411f0417 // indirect
- github.com/opencontainers/selinux v1.10.2 // indirect
+ github.com/opencontainers/runc v1.1.7 // indirect
+ github.com/opencontainers/runtime-spec v1.1.0-rc.3 // indirect
+ github.com/opencontainers/selinux v1.11.0 // indirect
github.com/ostreedev/ostree-go v0.0.0-20210805093236-719684c64e4f // indirect
github.com/pelletier/go-toml/v2 v2.0.8 // indirect
github.com/peterbourgon/diskv v2.0.1+incompatible // indirect
@@ -322,104 +321,105 @@ require (
github.com/pkg/sftp v1.13.5 // indirect
github.com/power-devops/perfstat v0.0.0-20210106213030-5aafc221ea8c // indirect
github.com/proglottis/gpgme v0.1.3 // indirect
- github.com/prometheus/client_golang v1.15.1 // indirect
+ github.com/prometheus/client_golang v1.16.0 // indirect
github.com/prometheus/client_model v0.4.0 // indirect
- github.com/prometheus/common v0.42.0 // indirect
- github.com/prometheus/procfs v0.9.0 // indirect
+ github.com/prometheus/common v0.44.0 // indirect
+ github.com/prometheus/procfs v0.10.1 // indirect
github.com/protocolbuffers/txtpbfmt v0.0.0-20230328191034-3462fbc510c0 // indirect
+ github.com/rancher/wharfie v0.6.2 // indirect
github.com/rcrowley/go-metrics v0.0.0-20201227073835-cf1acfcdf475 // indirect
github.com/rifflock/lfshook v0.0.0-20180920164130-b9218ef580f5 // indirect
- github.com/rivo/uniseg v0.4.3 // indirect
- github.com/rubenv/sql-migrate v1.2.0 // indirect
+ github.com/rivo/uniseg v0.4.4 // indirect
+ github.com/rubenv/sql-migrate v1.3.1 // indirect
github.com/segmentio/ksuid v1.0.4 // indirect
github.com/sergi/go-diff v1.2.0 // indirect
github.com/shoenig/go-m1cpu v0.1.6 // indirect
- github.com/shopspring/decimal v1.2.0 // indirect
- github.com/sigstore/fulcio v1.0.0 // indirect
- github.com/sigstore/rekor v1.2.0 // indirect
- github.com/sigstore/sigstore v1.6.4 // indirect
+ github.com/shopspring/decimal v1.3.1 // indirect
+ github.com/sigstore/fulcio v1.3.1 // indirect
+ github.com/sigstore/rekor v1.2.2-0.20230601122533-4c81ff246d12 // indirect
+ github.com/sigstore/sigstore v1.7.1 // indirect
github.com/skeema/knownhosts v1.1.0 // indirect
github.com/smartystreets/assertions v1.0.0 // indirect
github.com/soheilhy/cmux v0.1.5 // indirect
github.com/spf13/jwalterweatherman v1.1.0 // indirect
github.com/stefanberger/go-pkcs11uri v0.0.0-20201008174630-78d3cae3a980 // indirect
github.com/subosito/gotenv v1.4.2 // indirect
- github.com/sylabs/sif/v2 v2.9.0 // indirect
+ github.com/sylabs/sif/v2 v2.11.5 // indirect
github.com/syndtr/gocapability v0.0.0-20200815063812-42c35b437635 // indirect
- github.com/tchap/go-patricia v2.3.0+incompatible // indirect
+ github.com/tchap/go-patricia/v2 v2.3.1 // indirect
github.com/theupdateframework/go-tuf v0.5.2 // indirect
github.com/theupdateframework/notary v0.7.0 // indirect
github.com/titanous/rocacheck v0.0.0-20171023193734-afe73141d399 // indirect
github.com/tj/go-spin v1.1.0 // indirect
github.com/tklauser/go-sysconf v0.3.11 // indirect
github.com/tklauser/numcpus v0.6.0 // indirect
- github.com/tmc/grpc-websocket-proxy v0.0.0-20201229170055-e5319fda7802 // indirect
+ github.com/tmc/grpc-websocket-proxy v0.0.0-20220101234140-673ab2c3ae75 // indirect
github.com/ulikunitz/xz v0.5.11 // indirect
- github.com/vbatts/tar-split v0.11.2 // indirect
- github.com/vbauerster/mpb/v7 v7.5.3 // indirect
+ github.com/vbatts/tar-split v0.11.3 // indirect
+ github.com/vbauerster/mpb/v8 v8.4.0 // indirect
github.com/xanzy/ssh-agent v0.3.3 // indirect
github.com/xdg-go/pbkdf2 v1.0.0 // indirect
github.com/xdg-go/stringprep v1.0.4 // indirect
github.com/xeipuuv/gojsonpointer v0.0.0-20190905194746-02993c407bfb // indirect
github.com/xeipuuv/gojsonreference v0.0.0-20180127040603-bd5ef7bd5415 // indirect
github.com/xiang90/probing v0.0.0-20190116061207-43a291ad63a2 // indirect
- github.com/xlab/treeprint v1.1.0 // indirect
+ github.com/xlab/treeprint v1.2.0 // indirect
github.com/youmark/pkcs8 v0.0.0-20181117223130-1be2e3e5546d // indirect
github.com/yusufpapurcu/wmi v1.2.3 // indirect
github.com/zclconf/go-cty v1.12.1 // indirect
go.etcd.io/bbolt v1.3.7 // indirect
go.etcd.io/etcd/api/v3 v3.5.9 // indirect
go.etcd.io/etcd/client/pkg/v3 v3.5.9 // indirect
- go.etcd.io/etcd/client/v2 v2.305.8 // indirect
- go.etcd.io/etcd/etcdctl/v3 v3.5.6 // indirect
- go.etcd.io/etcd/pkg/v3 v3.5.8 // indirect
- go.etcd.io/etcd/raft/v3 v3.5.8 // indirect
- go.etcd.io/etcd/v3 v3.5.6 // indirect
+ go.etcd.io/etcd/client/v2 v2.305.9 // indirect
+ go.etcd.io/etcd/pkg/v3 v3.5.9 // indirect
+ go.etcd.io/etcd/raft/v3 v3.5.9 // indirect
go.mozilla.org/pkcs7 v0.0.0-20210826202110-33d05740a352 // indirect
go.opencensus.io v0.24.0 // indirect
- go.opentelemetry.io/contrib/instrumentation/google.golang.org/grpc/otelgrpc v0.35.0 // indirect
- go.opentelemetry.io/otel v1.14.0 // indirect
- go.opentelemetry.io/otel/exporters/otlp/internal/retry v1.11.2 // indirect
- go.opentelemetry.io/otel/exporters/otlp/otlptrace v1.11.2 // indirect
- go.opentelemetry.io/otel/exporters/otlp/otlptrace/otlptracegrpc v1.11.2 // indirect
- go.opentelemetry.io/otel/sdk v1.14.0 // indirect
- go.opentelemetry.io/otel/trace v1.14.0 // indirect
+ go.opentelemetry.io/contrib/instrumentation/google.golang.org/grpc/otelgrpc v0.40.0 // indirect
+ go.opentelemetry.io/otel v1.15.0 // indirect
+ go.opentelemetry.io/otel/exporters/otlp/internal/retry v1.15.0 // indirect
+ go.opentelemetry.io/otel/exporters/otlp/otlptrace v1.15.0 // indirect
+ go.opentelemetry.io/otel/exporters/otlp/otlptrace/otlptracegrpc v1.15.0 // indirect
+ go.opentelemetry.io/otel/metric v0.37.0 // indirect
+ go.opentelemetry.io/otel/sdk v1.15.0 // indirect
+ go.opentelemetry.io/otel/trace v1.15.0 // indirect
go.opentelemetry.io/proto/otlp v0.19.0 // indirect
- go.starlark.net v0.0.0-20201006213952-227f4aabceb5 // indirect
+ go.starlark.net v0.0.0-20230525235612-a134d8f9ddca // indirect
go.uber.org/atomic v1.10.0 // indirect
- go.uber.org/multierr v1.9.0 // indirect
- go4.org/intern v0.0.0-20211027215823-ae77deb06f29 // indirect
- go4.org/unsafe/assume-no-moving-gc v0.0.0-20230221090011-e4bae7ad2296 // indirect
+ go.uber.org/multierr v1.11.0 // indirect
+ go4.org/netipx v0.0.0-20230728184502-ec4c8b891b28 // indirect
golang.org/x/mod v0.11.0 // indirect
- golang.org/x/sys v0.11.0 // indirect
- golang.org/x/term v0.11.0 // indirect
+ golang.org/x/sys v0.12.0 // indirect
+ golang.org/x/term v0.12.0 // indirect
golang.org/x/time v0.3.0 // indirect
golang.org/x/tools v0.9.3 // indirect
golang.org/x/xerrors v0.0.0-20220907171357-04be3eba64a2 // indirect
- gomodules.xyz/jsonpatch/v2 v2.2.0 // indirect
- google.golang.org/api v0.122.0 // indirect
+ gomodules.xyz/jsonpatch/v2 v2.3.0 // indirect
+ google.golang.org/api v0.124.0 // indirect
google.golang.org/appengine v1.6.7 // indirect
- google.golang.org/genproto v0.0.0-20230410155749-daa745c078e1 // indirect
+ google.golang.org/genproto v0.0.0-20230526161137-0005af68ea54 // indirect
+ google.golang.org/genproto/googleapis/api v0.0.0-20230525234035-dd9d682886f9 // indirect
+ google.golang.org/genproto/googleapis/rpc v0.0.0-20230525234030-28d5490b6b19 // indirect
+ gopkg.in/go-jose/go-jose.v2 v2.6.1 // indirect
gopkg.in/ini.v1 v1.67.0 // indirect
- gopkg.in/natefinch/lumberjack.v2 v2.0.0 // indirect
+ gopkg.in/natefinch/lumberjack.v2 v2.2.1 // indirect
gopkg.in/square/go-jose.v2 v2.6.0 // indirect
gopkg.in/warnings.v0 v0.1.2 // indirect
gopkg.in/yaml.v3 v3.0.1 // indirect
- inet.af/netaddr v0.0.0-20220617031823-097006376321 // indirect
- k8s.io/apiserver v0.26.3 // indirect
- k8s.io/component-helpers v0.26.0 // indirect
- oras.land/oras-go v1.2.2 // indirect
+ k8s.io/apiserver v0.28.1 // indirect
+ k8s.io/component-helpers v0.28.2 // indirect
+ oras.land/oras-go v1.2.4 // indirect
periph.io/x/host/v3 v3.8.0 // indirect
sigs.k8s.io/json v0.0.0-20221116044647-bc3834ca7abd // indirect
- sigs.k8s.io/kustomize/api v0.12.1 // indirect
- sigs.k8s.io/kustomize/kustomize/v4 v4.5.7 // indirect
+ sigs.k8s.io/kustomize/api v0.13.5-0.20230601165947-6ce0bf390ce3 // indirect
+ sigs.k8s.io/kustomize/kustomize/v5 v5.0.4-0.20230601165947-6ce0bf390ce3 // indirect
sigs.k8s.io/structured-merge-diff/v4 v4.2.3 // indirect
)
replace (
- github.com/docker/cli => github.com/docker/cli v20.10.14+incompatible
+ github.com/docker/cli => github.com/docker/cli v24.0.6+incompatible
github.com/docker/distribution => github.com/docker/distribution v2.8.2+incompatible
- github.com/docker/docker => github.com/moby/moby v20.10.14+incompatible
+ github.com/docker/docker => github.com/moby/moby v24.0.6+incompatible
github.com/spf13/afero => github.com/spf13/afero v1.2.2
google.golang.org/grpc => google.golang.org/grpc v1.53.0
)
diff --git a/go.sum b/go.sum
index 046099b5d2f..8b6ca3de5fd 100644
--- a/go.sum
+++ b/go.sum
@@ -1,5 +1,4 @@
bazil.org/fuse v0.0.0-20160811212531-371fbbdaa898/go.mod h1:Xbm+BRKSBEpa4q4hTSxohYNQpsxXPbPry4JJWOB3LB8=
-bitbucket.org/creachadair/shell v0.0.7 h1:Z96pB6DkSb7F3Y3BBnJeOZH2gazyMTWlvecSD4vDqfk=
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.44.1/go.mod h1:iSa0KzasP4Uvy3f1mN/7PiObzGgflwredwwASm/v6AU=
@@ -119,8 +118,8 @@ cloud.google.com/go/compute v1.12.1/go.mod h1:e8yNOBcBONZU1vJKCvCoDw/4JQsA0dpM4x
cloud.google.com/go/compute v1.13.0/go.mod h1:5aPTS0cUNMIc1CE546K+Th6weJUNQErARyZtRXDJ8GE=
cloud.google.com/go/compute v1.14.0/go.mod h1:YfLtxrj9sU4Yxv+sXzZkyPjEyPBZfXHUvjxega5vAdo=
cloud.google.com/go/compute v1.15.1/go.mod h1:bjjoF/NtFUrkD/urWfdHaKuOPDR5nWIs63rR+SXhcpA=
-cloud.google.com/go/compute v1.19.0 h1:+9zda3WGgW1ZSTlVppLCYFIr48Pa35q1uG2N1itbCEQ=
-cloud.google.com/go/compute v1.19.0/go.mod h1:rikpw2y+UMidAe9tISo04EHNOIf42RLYF/q8Bs93scU=
+cloud.google.com/go/compute v1.19.3 h1:DcTwsFgGev/wV5+q8o2fzgcHOaac+DKGC91ZlvpsQds=
+cloud.google.com/go/compute v1.19.3/go.mod h1:qxvISKp/gYnXkSAD1ppcSOveRAmzxicEv/JlizULFrI=
cloud.google.com/go/compute/metadata v0.1.0/go.mod h1:Z1VN+bulIf6bt4P/C37K4DyZYZEXYonfTBHHFPO/4UU=
cloud.google.com/go/compute/metadata v0.2.0/go.mod h1:zFmK7XCadkQkj6TtorcaGlCW1hT1fIilQDwofLpJ20k=
cloud.google.com/go/compute/metadata v0.2.1/go.mod h1:jgHgmJd2RKBGzXqF5LR2EZMGxBkeanZ9wwa75XHJgOM=
@@ -230,7 +229,6 @@ cloud.google.com/go/lifesciences v0.6.0/go.mod h1:ddj6tSX/7BOnhxCSd3ZcETvtNr8NZ6
cloud.google.com/go/logging v1.6.1/go.mod h1:5ZO0mHHbvm8gEmeEUHrmDlTDSu5imF6MUP9OfilNXBw=
cloud.google.com/go/longrunning v0.1.1/go.mod h1:UUFxuDWkv22EuY93jjmDMFT5GPQKeFVJBIF6QlTqdsE=
cloud.google.com/go/longrunning v0.3.0/go.mod h1:qth9Y41RRSUE69rDcOn6DdK3HfQfsUI0YSmW3iIlLJc=
-cloud.google.com/go/longrunning v0.4.1 h1:v+yFJOfKC3yZdY6ZUI933pIYdhyhV8S3NpWrXWmg7jM=
cloud.google.com/go/managedidentities v1.3.0/go.mod h1:UzlW3cBOiPrzucO5qWkNkh0w33KFtBJU281hacNvsdE=
cloud.google.com/go/managedidentities v1.4.0/go.mod h1:NWSBYbEMgqmbZsLIyKvxrYbtqOsxY1ZrGM+9RgDqInM=
cloud.google.com/go/maps v0.1.0/go.mod h1:BQM97WGyfw9FWEmQMpZ5T6cpovXXSd1cGmFma94eubI=
@@ -396,100 +394,79 @@ cloud.google.com/go/workflows v1.8.0/go.mod h1:ysGhmEajwZxGn1OhGOGKsTXc5PyxOc0vf
cloud.google.com/go/workflows v1.9.0/go.mod h1:ZGkj1aFIOd9c8Gerkjjq7OW7I5+l6cSvT3ujaO/WwSA=
cuelang.org/go v0.6.0 h1:dJhgKCog+FEZt7OwAYV1R+o/RZPmE8aqFoptmxSWyr8=
cuelang.org/go v0.6.0/go.mod h1:9CxOX8aawrr3BgSdqPj7V0RYoXo7XIb+yDFC6uESrOQ=
+dario.cat/mergo v1.0.0 h1:AGCNq9Evsj31mOgNPcLyXc+4PNABt905YmuqPYYpBWk=
+dario.cat/mergo v1.0.0/go.mod h1:uNxQE+84aUszobStD9th8a29P2fMDhsBdgRYvZOxGmk=
dmitri.shuralyov.com/gpu/mtl v0.0.0-20190408044501-666a987793e9/go.mod h1:H6x//7gZCb22OMCxBHrMx7a5I7Hp++hsVxbQ4BYO7hU=
github.com/14rcole/gopopulate v0.0.0-20180821133914-b175b219e774 h1:SCbEWT58NSt7d2mcFdvxC9uyrdcTfvBbPLThhkDmXzg=
github.com/99designs/go-keychain v0.0.0-20191008050251-8e49817e8af4 h1:/vQbFIOMbk2FiG/kXiLl8BRyzTWDw7gX/Hz7Dd5eDMs=
github.com/99designs/go-keychain v0.0.0-20191008050251-8e49817e8af4/go.mod h1:hN7oaIRCjzsZ2dE+yG5k+rsdt3qcwykqK6HVGcKwsw4=
github.com/99designs/keyring v1.2.2 h1:pZd3neh/EmUzWONb35LxQfvuY7kiSXAq3HQd97+XBn0=
github.com/99designs/keyring v1.2.2/go.mod h1:wes/FrByc8j7lFOAGLGSNEg8f/PaI3cgTBqhFkHUrPk=
+github.com/AdaLogics/go-fuzz-headers v0.0.0-20230811130428-ced1acdcaa24 h1:bvDV9vkmnHYOMsOr4WLk+Vo07yKIzd94sVoIqshQ4bU=
+github.com/AdaLogics/go-fuzz-headers v0.0.0-20230811130428-ced1acdcaa24/go.mod h1:8o94RPi1/7XTJvwPpRSzSUedZrtlirdB3r9Z20bi2f8=
github.com/Azure/azure-pipeline-go v0.2.3 h1:7U9HBg1JFK3jHl5qmo4CTZKFTVgMwdFHMVtCdfBE21U=
github.com/Azure/azure-sdk-for-go v68.0.0+incompatible h1:fcYLmCpyNYRnvJbPerq7U0hS+6+I79yEDJBqVNcqUzU=
-github.com/Azure/azure-sdk-for-go/sdk/azcore v1.1.3 h1:8LoU8N2lIUzkmstvwXvVfniMZlFbesfT2AmA1aqvRr8=
-github.com/Azure/azure-sdk-for-go/sdk/azcore v1.1.3/go.mod h1:uGG2W01BaETf0Ozp+QxxKJdMBNRWPdstHG0Fmdwn1/U=
-github.com/Azure/azure-sdk-for-go/sdk/internal v1.0.0 h1:jp0dGvZ7ZK0mgqnTSClMxa5xuRL7NZgHameVYF6BurY=
-github.com/Azure/azure-sdk-for-go/sdk/internal v1.0.0/go.mod h1:eWRD7oawr1Mu1sLCawqVc0CUiF43ia3qQMxLscsKQ9w=
+github.com/Azure/azure-sdk-for-go/sdk/azcore v1.6.0 h1:8kDqDngH+DmVBiCtIjCFTGa7MBnsIOkF9IccInFEbjk=
+github.com/Azure/azure-sdk-for-go/sdk/internal v1.3.0 h1:sXr+ck84g/ZlZUOZiNELInmMgOsuGwdjjVkEIde0OtY=
github.com/Azure/azure-sdk-for-go/sdk/storage/azblob v0.3.0 h1:Px2UA+2RvSSvv+RvJNuUB6n7rs5Wsel4dXLe90Um2n4=
github.com/Azure/azure-storage-blob-go v0.14.0 h1:1BCg74AmVdYwO3dlKwtFU1V0wU2PZdREkXvAmZJRUlM=
github.com/Azure/go-ansiterm v0.0.0-20170929234023-d6e3b3328b78/go.mod h1:LmzpDX56iTiv29bbRTIsUNlaFfuhWRQBWjQdVyAevI8=
-github.com/Azure/go-ansiterm v0.0.0-20210617225240-d185dfc1b5a1 h1:UQHMgLO+TxOElx5B5HZ4hJQsoJ/PvUvKRhJHDQXO8P8=
-github.com/Azure/go-ansiterm v0.0.0-20210617225240-d185dfc1b5a1/go.mod h1:xomTg63KZ2rFqZQzSB4Vz2SUXa1BpHTVz9L5PTmPC4E=
+github.com/Azure/go-ansiterm v0.0.0-20230124172434-306776ec8161 h1:L/gRVlceqvL25UVaW/CKtUDjefjrs0SPonmDGUVOYP0=
+github.com/Azure/go-ansiterm v0.0.0-20230124172434-306776ec8161/go.mod h1:xomTg63KZ2rFqZQzSB4Vz2SUXa1BpHTVz9L5PTmPC4E=
github.com/Azure/go-autorest v14.2.0+incompatible h1:V5VMDjClD3GiElqLWO7mz2MxNAK/vTfRHdAubSIPRgs=
-github.com/Azure/go-autorest v14.2.0+incompatible/go.mod h1:r+4oMnoxhatjLLJ6zxSWATqVooLgysK6ZNox3g/xq24=
github.com/Azure/go-autorest/autorest v0.9.0/go.mod h1:xyHB1BMZT0cuDHU7I0+g046+BFDTQ8rEZB0s4Yfa6bI=
github.com/Azure/go-autorest/autorest v0.9.6/go.mod h1:/FALq9T/kS7b5J5qsQ+RSTUdAmGFqi0vUdVNNx8q630=
-github.com/Azure/go-autorest/autorest v0.11.1/go.mod h1:JFgpikqFJ/MleTTxwepExTKnFUKKszPS8UavbQYUMuw=
-github.com/Azure/go-autorest/autorest v0.11.29 h1:I4+HL/JDvErx2LjyzaVxllw2lRDB5/BT2Bm4g20iqYw=
+github.com/Azure/go-autorest/autorest v0.11.28 h1:ndAExarwr5Y+GaHE6VCaY1kyS/HwwGGyuimVhWsHOEM=
github.com/Azure/go-autorest/autorest/adal v0.5.0/go.mod h1:8Z9fGy2MpX0PvDjB1pEgQTmVqjGhiHBW7RJJEciWzS0=
github.com/Azure/go-autorest/autorest/adal v0.8.2/go.mod h1:ZjhuQClTqx435SRJ2iMlOxPYt3d2C/T/7TiQCVZSn3Q=
-github.com/Azure/go-autorest/autorest/adal v0.9.0/go.mod h1:/c022QCutn2P7uY+/oQWWNcK9YU+MH96NgK+jErpbcg=
-github.com/Azure/go-autorest/autorest/adal v0.9.5/go.mod h1:B7KF7jKIeC9Mct5spmyCB/A8CG/sEz1vwIRGv/bbw7A=
-github.com/Azure/go-autorest/autorest/adal v0.9.22 h1:/GblQdIudfEM3AWWZ0mrYJQSd7JS4S/Mbzh6F0ov0Xc=
-github.com/Azure/go-autorest/autorest/azure/auth v0.5.11 h1:P6bYXFoao05z5uhOQzbC3Qd8JqF3jUoocoTeIxkp2cA=
+github.com/Azure/go-autorest/autorest/adal v0.9.21 h1:jjQnVFXPfekaqb8vIsv2G1lxshoW+oGv4MDlhRtnYZk=
+github.com/Azure/go-autorest/autorest/azure/auth v0.5.12 h1:wkAZRgT/pn8HhFyzfe9UnqOjJYqlembgCTi72Bm/xKk=
github.com/Azure/go-autorest/autorest/azure/cli v0.4.6 h1:w77/uPk80ZET2F+AfQExZyEWtn+0Rk/uw17m9fv5Ajc=
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.2.0/go.mod h1:vcORJHLJEh643/Ioh9+vPmf1Ij9AEBM5FuBIXLmIy0g=
github.com/Azure/go-autorest/autorest/date v0.3.0 h1:7gUk1U5M/CQbp9WoqinNzJar+8KY+LPI6wiWrP/myHw=
-github.com/Azure/go-autorest/autorest/date v0.3.0/go.mod h1:BI0uouVdmngYNUzGWeSYnokU+TrmwEsOqdt8Y6sso74=
github.com/Azure/go-autorest/autorest/mocks v0.1.0/go.mod h1:OTyCOPRA2IgIlWxVYxBee2F5Gr4kF2zd2J5cFRaIDN0=
github.com/Azure/go-autorest/autorest/mocks v0.2.0/go.mod h1:OTyCOPRA2IgIlWxVYxBee2F5Gr4kF2zd2J5cFRaIDN0=
github.com/Azure/go-autorest/autorest/mocks v0.3.0/go.mod h1:a8FDP3DYzQ4RYfVAxAN3SVSiiO77gL2j2ronKKP0syM=
-github.com/Azure/go-autorest/autorest/mocks v0.4.0/go.mod h1:LTp+uSrOhSkaKrUy935gNZuuIPPVsHlr9DSOxSayd+k=
-github.com/Azure/go-autorest/autorest/mocks v0.4.1/go.mod h1:LTp+uSrOhSkaKrUy935gNZuuIPPVsHlr9DSOxSayd+k=
github.com/Azure/go-autorest/autorest/to v0.4.0 h1:oXVqrxakqqV1UZdSazDOPOLvOIz+XA683u8EctwboHk=
github.com/Azure/go-autorest/autorest/validation v0.3.1 h1:AgyqjAd94fwNAoTjl/WQXg4VvFeRFpO+UhNyRXqF1ac=
github.com/Azure/go-autorest/logger v0.1.0/go.mod h1:oExouG+K6PryycPJfVSxi/koC6LSNgds39diKLz7Vrc=
-github.com/Azure/go-autorest/logger v0.2.0/go.mod h1:T9E3cAhj2VqvPOtCYAvby9aBXkZmbF5NWuPV8+WeEW8=
github.com/Azure/go-autorest/logger v0.2.1 h1:IG7i4p/mDa2Ce4TRyAO8IHnVhAVF3RFU+ZtXWSmf4Tg=
github.com/Azure/go-autorest/tracing v0.5.0/go.mod h1:r/s2XiOKccPW3HrqB+W0TQzfbtp2fGCgRFtBroKn4Dk=
github.com/Azure/go-autorest/tracing v0.6.0 h1:TYi4+3m5t6K48TGI9AUdb+IzbnSxvnvUMfuitfgcfuo=
-github.com/Azure/go-autorest/tracing v0.6.0/go.mod h1:+vhtPC754Xsa23ID7GlGsrdKBpUA79WCAKPPZVC2DeU=
github.com/BurntSushi/toml v0.3.1/go.mod h1:xHWCNGjB5oqiDr8zfno3MHue2Ht5sIBksp03qcyfWMU=
-github.com/BurntSushi/toml v1.2.1 h1:9F2/+DoOYIOksmaJFPw1tGFy1eDnIJXg+UHjuD8lTak=
github.com/BurntSushi/toml v1.2.1/go.mod h1:CxXYINrC8qIiEnFrOxCa7Jy5BFHlXnUU2pbicEuybxQ=
+github.com/BurntSushi/toml v1.3.2 h1:o7IhLm0Msx3BaB+n3Ag7L8EVlByGnpq14C4YWiu/gL8=
+github.com/BurntSushi/toml v1.3.2/go.mod h1:CxXYINrC8qIiEnFrOxCa7Jy5BFHlXnUU2pbicEuybxQ=
github.com/BurntSushi/xgb v0.0.0-20160522181843-27f122750802/go.mod h1:IVnqGOEym/WlBOVXweHU+Q+/VP0lqqI8lqeDx9IjBqo=
github.com/DATA-DOG/go-sqlmock v1.5.0 h1:Shsta01QNfFxHCfpW6YH2STWB0MudeXXEWMr20OEh60=
github.com/DATA-DOG/go-sqlmock v1.5.0/go.mod h1:f/Ixk793poVmq4qj/V1dPUg2JEAKC73Q5eFN3EC/SaM=
github.com/MakeNowJust/heredoc v1.0.0 h1:cXCdzVdstXyiTqTvfqk9SDHpKNjxuom+DOlyEeQ4pzQ=
github.com/MakeNowJust/heredoc v1.0.0/go.mod h1:mG5amYoWBHf8vpLOuehzbGGw0EHxpZZ6lCpQ4fNJ8LE=
-github.com/Masterminds/goutils v1.1.0/go.mod h1:8cTjp+g8YejhMuvIA5y2vz3BpJxksy863GQaJW2MFNU=
github.com/Masterminds/goutils v1.1.1 h1:5nUrii3FMTL5diU80unEVvNevw1nH4+ZV4DSLVJLSYI=
github.com/Masterminds/goutils v1.1.1/go.mod h1:8cTjp+g8YejhMuvIA5y2vz3BpJxksy863GQaJW2MFNU=
github.com/Masterminds/semver/v3 v3.1.1/go.mod h1:VPu/7SZ7ePZ3QOrcuXROw5FAcLl4a0cBrbBpGY/8hQs=
github.com/Masterminds/semver/v3 v3.2.0/go.mod h1:qvl/7zhW3nngYb5+80sSMF+FG2BjYrf8m9wsX0PNOMQ=
github.com/Masterminds/semver/v3 v3.2.1 h1:RN9w6+7QoMeJVGyfmbcgs28Br8cvmnucEXnY0rYXWg0=
github.com/Masterminds/semver/v3 v3.2.1/go.mod h1:qvl/7zhW3nngYb5+80sSMF+FG2BjYrf8m9wsX0PNOMQ=
-github.com/Masterminds/sprig/v3 v3.2.0/go.mod h1:tWhwTbUTndesPNeF0C900vKoq283u6zp4APT9vaF3SI=
+github.com/Masterminds/sprig/v3 v3.2.1/go.mod h1:UoaO7Yp8KlPnJIYWTFkMaqPUYKTfGFPhxNuwnnxkKlk=
github.com/Masterminds/sprig/v3 v3.2.3 h1:eL2fZNezLomi0uOLqjQoN6BfsDD+fyLtgbJMAj9n6YA=
github.com/Masterminds/sprig/v3 v3.2.3/go.mod h1:rXcFaZ2zZbLRJv/xSysmlgIM1u11eBaRMhvYXJNkGuM=
-github.com/Masterminds/squirrel v1.5.3 h1:YPpoceAcxuzIljlr5iWpNKaql7hLeG1KLSrhvdHpkZc=
-github.com/Masterminds/squirrel v1.5.3/go.mod h1:NNaOrjSoIDfDA40n7sr2tPNZRfjzjA400rg+riTZj10=
-github.com/Microsoft/go-winio v0.4.11/go.mod h1:VhR8bwka0BXejwEJY73c50VrPtXAaKcyvVC4A4RozmA=
-github.com/Microsoft/go-winio v0.4.14/go.mod h1:qXqCSQ3Xa7+6tgxaGTIe4Kpcdsi+P8jBhyzoq1bpyYA=
-github.com/Microsoft/go-winio v0.4.15-0.20190919025122-fc70bd9a86b5/go.mod h1:tTuCMEN+UleMWgg9dVx4Hu52b1bJo+59jBh3ajtinzw=
+github.com/Masterminds/squirrel v1.5.4 h1:uUcX/aBc8O7Fg9kaISIUsHXdKuqehiXAMQTYX8afzqM=
+github.com/Masterminds/squirrel v1.5.4/go.mod h1:NNaOrjSoIDfDA40n7sr2tPNZRfjzjA400rg+riTZj10=
github.com/Microsoft/go-winio v0.4.16-0.20201130162521-d1ffc52c7331/go.mod h1:XB6nPKklQyQ7GC9LdcBEcBl8PF76WugXOPRXwdLnMv0=
github.com/Microsoft/go-winio v0.4.16/go.mod h1:XB6nPKklQyQ7GC9LdcBEcBl8PF76WugXOPRXwdLnMv0=
-github.com/Microsoft/go-winio v0.4.17-0.20210211115548-6eac466e5fa3/go.mod h1:JPGBdM1cNvN/6ISo+n8V5iA4v8pBzdOpzfwIujj1a84=
-github.com/Microsoft/go-winio v0.4.17-0.20210324224401-5516f17a5958/go.mod h1:JPGBdM1cNvN/6ISo+n8V5iA4v8pBzdOpzfwIujj1a84=
-github.com/Microsoft/go-winio v0.4.17/go.mod h1:JPGBdM1cNvN/6ISo+n8V5iA4v8pBzdOpzfwIujj1a84=
github.com/Microsoft/go-winio v0.5.2/go.mod h1:WpS1mjBmmwHBEWmogvA2mj8546UReBk4v8QkMxJ6pZY=
-github.com/Microsoft/go-winio v0.6.0 h1:slsWYD/zyx7lCXoZVlvQrj0hPTM1HI4+v1sIda2yDvg=
-github.com/Microsoft/go-winio v0.6.0/go.mod h1:cTAf44im0RAYeL23bpB+fzCyDH2MJiz2BO69KH/soAE=
-github.com/Microsoft/hcsshim v0.8.6/go.mod h1:Op3hHsoHPAvb6lceZHDtd9OkTew38wNoXnJs8iY7rUg=
-github.com/Microsoft/hcsshim v0.8.7-0.20190325164909-8abdbb8205e4/go.mod h1:Op3hHsoHPAvb6lceZHDtd9OkTew38wNoXnJs8iY7rUg=
-github.com/Microsoft/hcsshim v0.8.7/go.mod h1:OHd7sQqRFrYd3RmSgbgji+ctCwkbq2wbEYNSzOYtcBQ=
-github.com/Microsoft/hcsshim v0.8.9/go.mod h1:5692vkUqntj1idxauYlpoINNKeqCiG6Sg38RRsjT5y8=
+github.com/Microsoft/go-winio v0.6.1 h1:9/kr64B9VUZrLm5YYwbGtUJnMgqWVOdUAXu6Migciow=
+github.com/Microsoft/go-winio v0.6.1/go.mod h1:LRdKpFKfdobln8UmuiYcKPot9D2v6svN5+sAH+4kjUM=
github.com/Microsoft/hcsshim v0.8.14/go.mod h1:NtVKoYxQuTLx6gEq0L96c9Ju4JbRJ4nY2ow3VK6a9Lg=
-github.com/Microsoft/hcsshim v0.8.15/go.mod h1:x38A4YbHbdxJtc0sF6oIz+RG0npwSCAvn69iY6URG00=
-github.com/Microsoft/hcsshim v0.8.16/go.mod h1:o5/SZqmR7x9JNKsW3pu+nqHm0MF8vbA+VxGOoXdC600=
-github.com/Microsoft/hcsshim v0.8.21/go.mod h1:+w2gRZ5ReXQhFOrvSQeNfhrYB/dg3oDwTOcER2fw4I4=
-github.com/Microsoft/hcsshim v0.9.6 h1:VwnDOgLeoi2du6dAznfmspNqTiwczvjv4K7NxuY9jsY=
-github.com/Microsoft/hcsshim v0.9.6/go.mod h1:7pLA8lDk46WKDWlVsENo92gC0XFa8rbKfyFRBqxEbCc=
-github.com/Microsoft/hcsshim/test v0.0.0-20201218223536-d3e5debf77da/go.mod h1:5hlzMzRKMLyo42nCZ9oml8AdTlq/0cvIaBv6tK1RehU=
-github.com/Microsoft/hcsshim/test v0.0.0-20210227013316-43a75bb4edd3/go.mod h1:mw7qgWloBUl75W/gVH3cQszUg1+gUITj7D6NY7ywVnY=
+github.com/Microsoft/hcsshim v0.11.0 h1:7EFNIY4igHEXUdj1zXgAyU3fLc7QfOKHbkldRVTBdiM=
+github.com/Microsoft/hcsshim v0.11.0/go.mod h1:OEthFdQv/AD2RAdzR6Mm1N1KPCztGKDurW1Z8b8VGMM=
github.com/NYTimes/gziphandler v0.0.0-20170623195520-56545f4a5d46/go.mod h1:3wb06e3pkSAbeQ52E9H9iFoQsEEwGN64994WTCIhntQ=
github.com/OneOfOne/xxhash v1.2.2/go.mod h1:HSdplMjZKSmBqAxg5vPj2TmRDmfkzw+cTzAElWljhcU=
-github.com/ProtonMail/go-crypto v0.0.0-20230217124315-7d5c6f04bbb8 h1:wPbRQzjjwFc0ih8puEVAOFGELsn1zoIIYdxvML7mDxA=
github.com/ProtonMail/go-crypto v0.0.0-20230217124315-7d5c6f04bbb8/go.mod h1:I0gYDMZ6Z5GRU7l58bNFSkPTFN6Yl12dsUlAZ8xy98g=
+github.com/ProtonMail/go-crypto v0.0.0-20230528122434-6f98819771a1 h1:JMDGhoQvXNTqH6Y3MC0IUw6tcZvaUdujNqzK2HYWZc8=
+github.com/ProtonMail/go-crypto v0.0.0-20230528122434-6f98819771a1/go.mod h1:EjAoLdwvbIOoOQr3ihjnSoLZRtE8azugULFRteWMNc0=
github.com/PuerkitoBio/purell v1.0.0/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=
@@ -503,6 +480,7 @@ github.com/StudioSol/set v1.0.0 h1:G27J71la+Da08WidabBkoRrvPLTa4cdCn0RjvyJ5WKQ=
github.com/StudioSol/set v1.0.0/go.mod h1:hIUNZPo6rEGF43RlPXHq7Fjmf+HkVJBqAjtK7Z9LoIU=
github.com/VividCortex/ewma v1.2.0 h1:f58SaIzcDXrSy3kWaHNvuJgJ3Nmz59Zji6XoJR/q1ow=
github.com/VividCortex/ewma v1.2.0/go.mod h1:nz4BbCtbLyFDeC9SUHbtcT5644juEuWfUAUnGx7j5l4=
+github.com/a8m/expect v1.0.0/go.mod h1:4IwSCMumY49ScypDnjNbYEjgVeqy1/U2cEs3Lat96eA=
github.com/acarl005/stripansi v0.0.0-20180116102854-5a71ef0e047d h1:licZJFw2RwpHMqeKTCYkitsPqHNxTmd4SNR5r94FGM8=
github.com/acarl005/stripansi v0.0.0-20180116102854-5a71ef0e047d/go.mod h1:asat636LX7Bqt5lYEZ27JNDcqxfjdBQuJ/MM4CN/Lzo=
github.com/acomagu/bufpipe v1.0.4 h1:e3H4WUzM3npvo5uv95QuJM3cQspFNtFBzvJ2oNjKIDQ=
@@ -510,13 +488,9 @@ github.com/acomagu/bufpipe v1.0.4/go.mod h1:mxdxdup/WdsKVreO5GpW4+M/1CE2sMG4jeGJ
github.com/ahmetalpbalkan/go-cursor v0.0.0-20131010032410-8136607ea412 h1:vOVO0ypMfTt6tZacyI0kp+iCZb1XSNiYDqnzBWYgfe4=
github.com/ahmetalpbalkan/go-cursor v0.0.0-20131010032410-8136607ea412/go.mod h1:AI9hp1tkp10pAlK5TCwL+7yWbRgtDm9jhToq6qij2xs=
github.com/alecthomas/template v0.0.0-20160405071501-a0175ee3bccc/go.mod h1:LOuyumcjzFXgccqObfd/Ljyb9UuFJ6TxHnclSeseNhc=
-github.com/alecthomas/template v0.0.0-20190718012654-fb15b899a751/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-20190717042225-c3de453c63f4/go.mod h1:ybxpYRFXyAe+OPACYpWeL0wqObRcbAqCMya13uyzqw0=
-github.com/alecthomas/units v0.0.0-20190924025748-f65c72e2690d/go.mod h1:rBZYJk541a8SKzHPHnH3zbiI+7dagKZ0cgpgrD7Fyho=
github.com/alecthomas/units v0.0.0-20211218093645-b94a6e3cc137 h1:s6gZFSlWYmbqAuRjVTiNNhvNRfY2Wxp9nhfyel4rklc=
github.com/alecthomas/units v0.0.0-20211218093645-b94a6e3cc137/go.mod h1:OMCwj8VM1Kc9e19TLln2VL61YJF0x1XFtfdL4JdbSyE=
-github.com/alexflint/go-filemutex v0.0.0-20171022225611-72bdc8eae2ae/go.mod h1:CgnQgUtFrFz9mxFNtED3jI5tLDjKlOM+oUF/sTk6ps0=
github.com/anmitsu/go-shlex v0.0.0-20200514113438-38f4b401e2be h1:9AeTilPcZAjCFIImctFaOjnTIavg87rW78vTPkQqLI8=
github.com/anmitsu/go-shlex v0.0.0-20200514113438-38f4b401e2be/go.mod h1:ySMOLuWl6zY27l47sB3qLNK6tF2fkHG55UZxx8oIVo4=
github.com/antihax/optional v1.0.0/go.mod h1:uupD/76wgC+ih3iEmQUL+0Ugr19nfwCT1kdvxnR2qWY=
@@ -528,9 +502,9 @@ github.com/armon/circbuf v0.0.0-20150827004946-bbbad097214e/go.mod h1:3U/XgcO3hC
github.com/armon/consul-api v0.0.0-20180202201655-eb2c6b5be1b6/go.mod h1:grANhF5doyWs3UAsr3K4I6qtAmlQcZDesFNEHPZAzj8=
github.com/armon/go-metrics v0.0.0-20180917152333-f0300d1749da/go.mod h1:Q73ZrmVTwzkszR9V5SSuryQ31EELlFMUz1kKyl939pY=
github.com/armon/go-radix v0.0.0-20180808171621-7fddfc383310/go.mod h1:ufUuZ+zHj4x4TnLV4JWEpy2hxWSpsRywHrMgIH9cCH8=
+github.com/armon/go-radix v1.0.0/go.mod h1:ufUuZ+zHj4x4TnLV4JWEpy2hxWSpsRywHrMgIH9cCH8=
github.com/armon/go-socks5 v0.0.0-20160902184237-e75332964ef5 h1:0CwZNZbxp69SHPdPJAN/hZIm0C4OItdklCFmMRWYpio=
github.com/armon/go-socks5 v0.0.0-20160902184237-e75332964ef5/go.mod h1:wHh0iHkYZB8zMSxRWpUBQtwG5a7fFgvEO+odwuTv2gs=
-github.com/asaskevich/govalidator v0.0.0-20190424111038-f61b66f89f4a/go.mod h1:lB+ZfQJz7igIIfQNfa7Ml4HSf2uFQQRzpGGRXenZAgY=
github.com/asaskevich/govalidator v0.0.0-20200907205600-7a23bdc65eef/go.mod h1:WaHUgvxTVq04UNunO+XhnAqY/wQc+bxr74GqbsZ/Jqw=
github.com/asaskevich/govalidator v0.0.0-20230301143203-a9d515a09cc2 h1:DklsrG3dyBCFEj5IhUbnKptjxatkF07cF2ak3yi77so=
github.com/asaskevich/govalidator v0.0.0-20230301143203-a9d515a09cc2/go.mod h1:WaHUgvxTVq04UNunO+XhnAqY/wQc+bxr74GqbsZ/Jqw=
@@ -549,7 +523,6 @@ github.com/beorn7/perks v1.0.1 h1:VlbKKnNfV8bJzeqoa4cOKqO6bYr3WgKZxO8Z16+hsOM=
github.com/beorn7/perks v1.0.1/go.mod h1:G2ZrVWU2WbWT9wwq4/hrbKbnv/1ERSJQ0ibhJ6rlkpw=
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/bgentry/speakeasy v0.1.0 h1:ByYyxL9InA1OWqxJqqp2A5pYHUrCiAL6K3J+LKSsQkY=
github.com/bgentry/speakeasy v0.1.0/go.mod h1:+zsyZBPWlz7T6j88CTgSN5bM796AkVf0kBD4zp0CCIs=
github.com/bhmj/jsonslice v1.1.2 h1:Lzen2S9iG3HsESpiIAnTM7Obs1QiTz83ZXa5YrpTTWI=
github.com/bhmj/jsonslice v1.1.2/go.mod h1:O3ZoA0zdEefdbk1dkU5aWPOA36zQhhS/HV6RQFLTlnU=
@@ -557,11 +530,8 @@ github.com/bhmj/xpression v0.9.1 h1:N7bX/nWx9oFi/zsiMTx2ehoRApTDAWdQadq/5o2wMGk=
github.com/bhmj/xpression v0.9.1/go.mod h1:j9oYmEXJjeL9mrgW1+ZDBKJXnbupsCPGhlO9J5YhS1Q=
github.com/bitly/go-hostpool v0.1.0/go.mod h1:4gOCgp6+NZnVqlKyZ/iBZFTAJKembaVENUpMkpg42fw=
github.com/bitly/go-simplejson v0.5.0/go.mod h1:cXHtHw4XUPsvGaxgjIAn8PhEWG9NfngEKAMDJEczWVA=
-github.com/bits-and-blooms/bitset v1.2.0/go.mod h1:gIdJ4wp64HaoK2YrL1Q5/N7Y16edYb8uY+O0FJTyyDA=
github.com/bketelsen/crypt v0.0.3-0.20200106085610-5cbc8cc4026c/go.mod h1:MKsuJmJgSg28kpZDP6UIiPt0e0Oz0kqKNGyRaWEPv84=
github.com/bketelsen/crypt v0.0.4/go.mod h1:aI6NrJ0pMGgvZKL1iVgXLnfIFJtfV+bKCoqOes/6LfM=
-github.com/blang/semver v3.1.0+incompatible/go.mod h1:kRBLl5iJ+tD4TcOOxsy/0fnwebNt5EWlYSAyrTnjyyk=
-github.com/blang/semver v3.5.1+incompatible/go.mod h1:kRBLl5iJ+tD4TcOOxsy/0fnwebNt5EWlYSAyrTnjyyk=
github.com/blang/semver/v4 v4.0.0 h1:1PFHFE6yCCTv8C1TeyNNarDzntLi7wMI5i/pzqYIsAM=
github.com/blang/semver/v4 v4.0.0/go.mod h1:IbckMUScFkM3pff0VJDNKRiT6TG/YpiHIM2yvyW5YoQ=
github.com/bmizerany/assert v0.0.0-20160611221934-b7ed37b82869/go.mod h1:Ekp36dRnpXw/yCqJaO+ZrUyxD+3VXMFFr56k5XYrpB4=
@@ -570,37 +540,29 @@ github.com/briandowns/spinner v1.23.0/go.mod h1:rPG4gmXeN3wQV/TsAY4w8lPdIM6RX3yq
github.com/bshuster-repo/logrus-logstash-hook v1.0.2 h1:JYRWo+QGnQdedgshosug9hxpPYTB9oJ1ZZD3fY31alU=
github.com/bsm/ginkgo/v2 v2.7.0 h1:ItPMPH90RbmZJt5GtkcNvIRuGEdwlBItdNVoyzaNQao=
github.com/bsm/gomega v1.26.0 h1:LhQm+AFcgV2M0WyKroMASzAzCAJVpAxQXv4SaI9a69Y=
-github.com/buger/jsonparser v0.0.0-20180808090653-f4dd9f5a6b44/go.mod h1:bbYlZJ7hK1yFx9hf58LP0zeX7UjIGs20ufpu3evjr+s=
-github.com/buger/jsonparser v1.1.1/go.mod h1:6RYKKt7H4d4+iWqouImQ9R2FZql3VbhNgx27UK13J/0=
github.com/bugsnag/bugsnag-go v1.0.5-0.20150529004307-13fd6b8acda0/go.mod h1:2oa8nejYd4cQ/b0hMIopN0lCRxU0bueqREvZLWFrtK8=
github.com/bugsnag/bugsnag-go v2.1.2+incompatible h1:E7dor84qzwUO8KdCM68CZwq9QOSR7HXlLx3Wj5vui2s=
github.com/bugsnag/osext v0.0.0-20130617224835-0dd3f918b21b/go.mod h1:obH5gd0BsqsP2LwDJ9aOkm/6J86V6lyAXCoQWGw3K50=
github.com/bugsnag/panicwrap v0.0.0-20151223152923-e2c28503fcd0/go.mod h1:D/8v3kj0zr8ZAKg1AQ6crr+5VwKN5eIywRkfhyM/+dE=
github.com/bugsnag/panicwrap v1.3.4 h1:A6sXFtDGsgU/4BLf5JT0o5uYg3EeKgGx3Sfs+/uk3pU=
github.com/bwesterb/go-ristretto v1.2.0/go.mod h1:fUIoIZaG73pV5biE2Blr2xEzDoMj7NFEuV9ekS419A0=
+github.com/bwesterb/go-ristretto v1.2.3/go.mod h1:fUIoIZaG73pV5biE2Blr2xEzDoMj7NFEuV9ekS419A0=
github.com/bxcodec/faker v2.0.1+incompatible h1:P0KUpUw5w6WJXwrPfv35oc91i4d8nf40Nwln+M/+faA=
github.com/c9s/goprocinfo v0.0.0-20170724085704-0010a05ce49f h1:tRk+aBit+q3oqnj/1mF5HHhP2yxJM2lSa0afOJxQ3nE=
github.com/c9s/goprocinfo v0.0.0-20170724085704-0010a05ce49f/go.mod h1:uEyr4WpAH4hio6LFriaPkL938XnrvLpNPmQHBdrmbIE=
-github.com/cenkalti/backoff/v4 v4.1.1/go.mod h1:scbssz8iZGpm3xbr14ovlUdkxfGXNInqkPWOWmG2CLw=
github.com/cenkalti/backoff/v4 v4.2.1 h1:y4OZtCnogmCPw98Zjyt5a6+QwPLGkiQsYW5oUqylYbM=
github.com/cenkalti/backoff/v4 v4.2.1/go.mod h1:Y3VNntkOUPxTVeUxJ/G5vcM//AlwfmyYozVcomhLiZE=
github.com/census-instrumentation/opencensus-proto v0.2.1/go.mod h1:f6KPmirojxKA12rnyqOA5BBL4O983OfeGPqjHWSTneU=
github.com/census-instrumentation/opencensus-proto v0.3.0/go.mod h1:f6KPmirojxKA12rnyqOA5BBL4O983OfeGPqjHWSTneU=
-github.com/census-instrumentation/opencensus-proto v0.4.1 h1:iKLQ0xPNFxR/2hzXZMrBo8f1j86j5WHzznCCQxV/b8g=
github.com/census-instrumentation/opencensus-proto v0.4.1/go.mod h1:4T9NM4+4Vw91VeyqjLS6ao50K5bOcLKN6Q42XnYaRYw=
-github.com/certifi/gocertifi v0.0.0-20191021191039-0944d244cd40/go.mod h1:sGbDF6GwGcLpkNXPUTkMRoywsNa/ol15pxFe6ERfguA=
-github.com/certifi/gocertifi v0.0.0-20200922220541-2c3bb06c6054/go.mod h1:sGbDF6GwGcLpkNXPUTkMRoywsNa/ol15pxFe6ERfguA=
github.com/cespare/xxhash v1.1.0/go.mod h1:XrSqR1VqqWfGrhpAt58auRo0WTKS1nRRg3ghfAqPWnc=
github.com/cespare/xxhash/v2 v2.1.1/go.mod h1:VGX0DQ3Q6kWi7AoAeZDth3/j3BFtOZR5XLFGgcrjCOs=
github.com/cespare/xxhash/v2 v2.2.0 h1:DC2CZ1Ep5Y4k3ZQ899DldepgrayRUGE6BBZ/cd9Cj44=
github.com/cespare/xxhash/v2 v2.2.0/go.mod h1:VGX0DQ3Q6kWi7AoAeZDth3/j3BFtOZR5XLFGgcrjCOs=
github.com/chai2010/gettext-go v1.0.2 h1:1Lwwip6Q2QGsAdl/ZKPCwTe9fe0CjlUbqj5bFNSjIRk=
github.com/chai2010/gettext-go v1.0.2/go.mod h1:y+wnP2cHYaVj19NZhYKAwEMH2CI1gNHeQQ+5AjwawxA=
-github.com/chaos-mesh/chaos-mesh/api v0.0.0-20230423031423-0b31a519b502 h1:dlu7F5rX2PA4laECDbFXwtDKktUK31lcC09wU70L3QY=
-github.com/chaos-mesh/chaos-mesh/api v0.0.0-20230423031423-0b31a519b502/go.mod h1:5qllHIhMkPEWjIimDum42JtMj0P1Tn9x91XUceuPNjY=
-github.com/checkpoint-restore/go-criu/v4 v4.1.0/go.mod h1:xUQBLp4RLc5zJtWY++yjOoMoB5lihDt7fai+75m+rGw=
-github.com/checkpoint-restore/go-criu/v5 v5.0.0/go.mod h1:cfwC0EG7HMUenopBsUf9d89JlCLQIfgVcNsNN0t6T2M=
-github.com/checkpoint-restore/go-criu/v5 v5.3.0/go.mod h1:E/eQpaFtUKGOOSEBZgmKAcn+zUUwWxqcaKZlF54wK8E=
+github.com/chaos-mesh/chaos-mesh/api v0.0.0-20230912020346-a5d89c1c90ad h1:DVxCvjXlmkm4idu4bAbI9P+D99BsVHTKOKbzRYTlFwU=
+github.com/chaos-mesh/chaos-mesh/api v0.0.0-20230912020346-a5d89c1c90ad/go.mod h1:Yi/tSmvDrnFgyZN4bsXm3gfXrp3zo1uytHmnPEYfquM=
github.com/cheggaaa/pb v1.0.27/go.mod h1:pQciLPpbU0oxA0h+VJYYLxO+XeDQb5pZijXscXHm81s=
github.com/chmduquesne/rollinghash v4.0.0+incompatible h1:hnREQO+DXjqIw3rUTzWN7/+Dpw+N5Um8zpKV0JOEgbo=
github.com/chzyer/logex v1.1.10/go.mod h1:+Ywpsq7O8HXn0nuIou7OrIPyXbp3wmkHB+jjWRnGsAI=
@@ -613,11 +575,6 @@ github.com/chzyer/test v0.0.0-20180213035817-a1ea475d72b1/go.mod h1:Q3SI9o4m/ZMn
github.com/chzyer/test v1.0.0 h1:p3BQDXSxOhOG0P9z6/hGnII4LGiEPOYBhs8asl/fC04=
github.com/chzyer/test v1.0.0/go.mod h1:2JlltgoNkt4TW/z9V/IzDdFaMTM2JPIi26O1pF38GC8=
github.com/cilium/ebpf v0.0.0-20200110133405-4032b1d8aae3/go.mod h1:MA5e5Lr8slmEg9bt0VpxxWqJlO4iwu3FBdHUzV7wQVg=
-github.com/cilium/ebpf v0.0.0-20200702112145-1c8d4c9ef775/go.mod h1:7cR51M8ViRLIdUjrmSXlK9pkrsDlLHbO8jiB8X8JnOc=
-github.com/cilium/ebpf v0.2.0/go.mod h1:To2CFviqOWL/M0gIMsvSMlqe7em/l1ALkX1PyjrX2Qs=
-github.com/cilium/ebpf v0.4.0/go.mod h1:4tRaxcgiL706VnOzHOdBlY8IEAIdxINsQBcU4xJJXRs=
-github.com/cilium/ebpf v0.6.2/go.mod h1:4tRaxcgiL706VnOzHOdBlY8IEAIdxINsQBcU4xJJXRs=
-github.com/cilium/ebpf v0.7.0/go.mod h1:/oI2+1shJiTGAMgl6/RgJr36Eo1jzrRcAWbcXO2usCA=
github.com/clbanning/mxj/v2 v2.5.7 h1:7q5lvUpaPF/WOkqgIDiwjBJaznaLCCBd78pi8ZyAnE0=
github.com/clbanning/mxj/v2 v2.5.7/go.mod h1:hNiWqW14h+kc+MdF9C6/YoRfjEJoR3ou6tn/Qo+ve2s=
github.com/cloudflare/cfssl v0.0.0-20180223231731-4e2dcbde5004 h1:lkAMpLVBDaj17e85keuznYcH5rqI438v41pKcBl4ZxQ=
@@ -626,161 +583,68 @@ github.com/cloudflare/circl v1.1.0/go.mod h1:prBCrKB9DV4poKZY1l9zBXg2QJY7mvgRvtM
github.com/cloudflare/circl v1.3.3 h1:fE/Qz0QdIGqeWfnwq0RE0R7MI51s0M2E4Ga9kq5AEMs=
github.com/cloudflare/circl v1.3.3/go.mod h1:5XYMA4rFBvNIrhs50XuiBJ15vF2pZn4nnUKZrLbUZFA=
github.com/cncf/udpa/go v0.0.0-20210930031921-04548b0d99d4/go.mod h1:6pvJx4me5XPnfI9Z40ddWsdw2W/uZgQLFXToKeRcDiI=
-github.com/cncf/udpa/go v0.0.0-20220112060539-c52dc94e7fbe h1:QQ3GSy+MqSHxm/d8nCtnAiZdYFd45cYZPs8vOOIYKfk=
github.com/cncf/udpa/go v0.0.0-20220112060539-c52dc94e7fbe/go.mod h1:6pvJx4me5XPnfI9Z40ddWsdw2W/uZgQLFXToKeRcDiI=
github.com/cncf/xds/go v0.0.0-20210922020428-25de7278fc84/go.mod h1:eXthEFrGJvWHgFFCl3hGmgk+/aYT6PnTQLykKQRLhEs=
github.com/cncf/xds/go v0.0.0-20211001041855-01bcc9b48dfe/go.mod h1:eXthEFrGJvWHgFFCl3hGmgk+/aYT6PnTQLykKQRLhEs=
github.com/cncf/xds/go v0.0.0-20211011173535-cb28da3451f1/go.mod h1:eXthEFrGJvWHgFFCl3hGmgk+/aYT6PnTQLykKQRLhEs=
github.com/cncf/xds/go v0.0.0-20220314180256-7f1daf1720fc/go.mod h1:eXthEFrGJvWHgFFCl3hGmgk+/aYT6PnTQLykKQRLhEs=
-github.com/cncf/xds/go v0.0.0-20230105202645-06c439db220b h1:ACGZRIr7HsgBKHsueQ1yM4WaVaXh21ynwqsF8M8tXhA=
github.com/cncf/xds/go v0.0.0-20230105202645-06c439db220b/go.mod h1:eXthEFrGJvWHgFFCl3hGmgk+/aYT6PnTQLykKQRLhEs=
github.com/cockroachdb/apd/v3 v3.2.0 h1:79kHCn4tO0VGu3W0WujYrMjBDk8a2H4KEUYcXf7whcg=
github.com/cockroachdb/apd/v3 v3.2.0/go.mod h1:klXJcjp+FffLTHlhIG69tezTDvdP065naDsHzKhYSqc=
-github.com/cockroachdb/datadriven v0.0.0-20190809214429-80d97fb3cbaa/go.mod h1:zn76sxSg3SzpJ0PPJaLDCu+Bu0Lg3sKTORVIj19EIF8=
-github.com/cockroachdb/datadriven v0.0.0-20200714090401-bf6692d28da5/go.mod h1:h6jFvWxBdQXxjopDMZyH2UVceIRfR84bdzbkoKrsWNo=
github.com/cockroachdb/datadriven v1.0.2 h1:H9MtNqVoVhvd9nCBwOyDjUEdZCREqbIdCJD93PBm/jA=
-github.com/cockroachdb/errors v1.2.4/go.mod h1:rQD95gz6FARkaKkQXUksEje/d9a6wBJoCr5oaCLELYA=
-github.com/cockroachdb/logtags v0.0.0-20190617123548-eb05cc24525f/go.mod h1:i/u985jwjWRlyHXQbwatDASoW0RMlZ/3i9yJHE2xLkI=
-github.com/containerd/aufs v0.0.0-20200908144142-dab0cbea06f4/go.mod h1:nukgQABAEopAHvB6j7cnP5zJ+/3aVcE7hCYqvIwAHyE=
-github.com/containerd/aufs v0.0.0-20201003224125-76a6863f2989/go.mod h1:AkGGQs9NM2vtYHaUen+NljV0/baGCAPELGm2q9ZXpWU=
-github.com/containerd/aufs v0.0.0-20210316121734-20793ff83c97/go.mod h1:kL5kd6KM5TzQjR79jljyi4olc1Vrx6XBlcyj3gNv2PU=
-github.com/containerd/aufs v1.0.0/go.mod h1:kL5kd6KM5TzQjR79jljyi4olc1Vrx6XBlcyj3gNv2PU=
-github.com/containerd/btrfs v0.0.0-20201111183144-404b9149801e/go.mod h1:jg2QkJcsabfHugurUvvPhS3E08Oxiuh5W/g1ybB4e0E=
-github.com/containerd/btrfs v0.0.0-20210316141732-918d888fb676/go.mod h1:zMcX3qkXTAi9GI50+0HOeuV8LU2ryCE/V2vG/ZBiTss=
-github.com/containerd/btrfs v1.0.0/go.mod h1:zMcX3qkXTAi9GI50+0HOeuV8LU2ryCE/V2vG/ZBiTss=
-github.com/containerd/cgroups v0.0.0-20190717030353-c4b9ac5c7601/go.mod h1:X9rLEHIqSf/wfK8NsPqxJmeZgW4pcfzdXITDrUSJ6uI=
-github.com/containerd/cgroups v0.0.0-20190919134610-bf292b21730f/go.mod h1:OApqhQ4XNSNC13gXIwDjhOQxjWa/NxkwZXJ1EvqT0ko=
github.com/containerd/cgroups v0.0.0-20200531161412-0dbf7f05ba59/go.mod h1:pA0z1pT8KYB3TCXK/ocprsh7MAkoW8bZVzPdih9snmM=
-github.com/containerd/cgroups v0.0.0-20200710171044-318312a37340/go.mod h1:s5q4SojHctfxANBDvMeIaIovkq29IP48TKAxnhYRxvo=
-github.com/containerd/cgroups v0.0.0-20200824123100-0b889c03f102/go.mod h1:s5q4SojHctfxANBDvMeIaIovkq29IP48TKAxnhYRxvo=
-github.com/containerd/cgroups v0.0.0-20210114181951-8a68de567b68/go.mod h1:ZJeTFisyysqgcCdecO57Dj79RfL0LNeGiFUqLYQRYLE=
-github.com/containerd/cgroups v1.0.1/go.mod h1:0SJrPIenamHDcZhEcJMNBB85rHcUsw4f25ZfBiPYRkU=
-github.com/containerd/cgroups v1.0.4 h1:jN/mbWBEaz+T1pi5OFtnkQ+8qnmEbAr1Oo1FRm5B0dA=
-github.com/containerd/cgroups v1.0.4/go.mod h1:nLNQtsF7Sl2HxNebu77i1R0oDlhiTG+kO4JTrUzo6IA=
+github.com/containerd/cgroups v1.1.0 h1:v8rEWFl6EoqHB+swVNjVoCJE8o3jX7e8nqBGPLaDFBM=
+github.com/containerd/cgroups v1.1.0/go.mod h1:6ppBcbh/NOOUU+dMKrykgaBnK9lCIBxHqJDGwsa1mIw=
github.com/containerd/console v0.0.0-20180822173158-c12b1e7919c1/go.mod h1:Tj/on1eG8kiEhd0+fhSDzsPAFESxzBBvdyEgyryXffw=
-github.com/containerd/console v0.0.0-20181022165439-0650fd9eeb50/go.mod h1:Tj/on1eG8kiEhd0+fhSDzsPAFESxzBBvdyEgyryXffw=
-github.com/containerd/console v0.0.0-20191206165004-02ecf6a7291e/go.mod h1:8Pf4gM6VEbTNRIT26AyyU7hxdQU3MvAvxVI0sc00XBE=
-github.com/containerd/console v1.0.1/go.mod h1:XUsP6YE/mKtz6bxc+I8UiKKTP04qjQL4qcS3XoQ5xkw=
-github.com/containerd/console v1.0.2/go.mod h1:ytZPjGgY2oeTkAONYafi2kSj0aYggsf8acV1PGKCbzQ=
-github.com/containerd/console v1.0.3/go.mod h1:7LqA/THxQ86k76b8c/EMSiaJ3h1eZkMkXar0TQ1gf3U=
-github.com/containerd/containerd v1.2.10/go.mod h1:bC6axHOhabU15QhwfG7w5PipXdVtMXFTttgp+kVtyUA=
-github.com/containerd/containerd v1.3.0-beta.2.0.20190828155532-0293cbd26c69/go.mod h1:bC6axHOhabU15QhwfG7w5PipXdVtMXFTttgp+kVtyUA=
-github.com/containerd/containerd v1.3.0/go.mod h1:bC6axHOhabU15QhwfG7w5PipXdVtMXFTttgp+kVtyUA=
-github.com/containerd/containerd v1.3.1-0.20191213020239-082f7e3aed57/go.mod h1:bC6axHOhabU15QhwfG7w5PipXdVtMXFTttgp+kVtyUA=
github.com/containerd/containerd v1.3.2/go.mod h1:bC6axHOhabU15QhwfG7w5PipXdVtMXFTttgp+kVtyUA=
-github.com/containerd/containerd v1.4.0-beta.2.0.20200729163537-40b22ef07410/go.mod h1:bC6axHOhabU15QhwfG7w5PipXdVtMXFTttgp+kVtyUA=
-github.com/containerd/containerd v1.4.1/go.mod h1:bC6axHOhabU15QhwfG7w5PipXdVtMXFTttgp+kVtyUA=
github.com/containerd/containerd v1.4.3/go.mod h1:bC6axHOhabU15QhwfG7w5PipXdVtMXFTttgp+kVtyUA=
-github.com/containerd/containerd v1.5.0-beta.1/go.mod h1:5HfvG1V2FsKesEGQ17k5/T7V960Tmcumvqn8Mc+pCYQ=
-github.com/containerd/containerd v1.5.0-beta.3/go.mod h1:/wr9AVtEM7x9c+n0+stptlo/uBBoBORwEx6ardVcmKU=
-github.com/containerd/containerd v1.5.0-beta.4/go.mod h1:GmdgZd2zA2GYIBZ0w09ZvgqEq8EfBp/m3lcVZIvPHhI=
-github.com/containerd/containerd v1.5.0-rc.0/go.mod h1:V/IXoMqNGgBlabz3tHD2TWDoTJseu1FGOKuoA4nNb2s=
-github.com/containerd/containerd v1.5.1/go.mod h1:0DOxVqwDy2iZvrZp2JUx/E+hS0UNTVn7dJnIOwtYR4g=
-github.com/containerd/containerd v1.5.7/go.mod h1:gyvv6+ugqY25TiXxcZC3L5yOeYgEw0QMhscqVp1AR9c=
-github.com/containerd/containerd v1.6.18 h1:qZbsLvmyu+Vlty0/Ex5xc0z2YtKpIsb5n45mAMI+2Ns=
-github.com/containerd/containerd v1.6.18/go.mod h1:1RdCUu95+gc2v9t3IL+zIlpClSmew7/0YS8O5eQZrOw=
+github.com/containerd/containerd v1.7.6 h1:oNAVsnhPoy4BTPQivLgTzI9Oleml9l/+eYIDYXRCYo8=
+github.com/containerd/containerd v1.7.6/go.mod h1:SY6lrkkuJT40BVNO37tlYTSnKJnP5AXBc0fhx0q+TJ4=
github.com/containerd/continuity v0.0.0-20190426062206-aaeac12a7ffc/go.mod h1:GL3xCUCBDV3CZiTSEKksMWbLE66hEyuu9qyDOOqM47Y=
-github.com/containerd/continuity v0.0.0-20190815185530-f2a389ac0a02/go.mod h1:GL3xCUCBDV3CZiTSEKksMWbLE66hEyuu9qyDOOqM47Y=
-github.com/containerd/continuity v0.0.0-20191127005431-f65d91d395eb/go.mod h1:GL3xCUCBDV3CZiTSEKksMWbLE66hEyuu9qyDOOqM47Y=
-github.com/containerd/continuity v0.0.0-20200710164510-efbc4488d8fe/go.mod h1:cECdGN1O8G9bgKTlLhuPJimka6Xb/Gg7vYzCTNVxhvo=
github.com/containerd/continuity v0.0.0-20201208142359-180525291bb7/go.mod h1:kR3BEg7bDFaEddKm54WSmrol1fKWDU1nKYkgrcgZT7Y=
-github.com/containerd/continuity v0.0.0-20210208174643-50096c924a4e/go.mod h1:EXlVlkqNba9rJe3j7w3Xa924itAMLgZH4UD/Q4PExuQ=
-github.com/containerd/continuity v0.1.0/go.mod h1:ICJu0PwR54nI0yPEnJ6jcS+J7CZAUXrLh8lPo2knzsM=
-github.com/containerd/continuity v0.3.0 h1:nisirsYROK15TAMVukJOUyGJjz4BNQJBVsNvAXZJ/eg=
-github.com/containerd/fifo v0.0.0-20180307165137-3d5202aec260/go.mod h1:ODA38xgv3Kuk8dQz2ZQXpnv/UZZUHUCL7pnLehbXgQI=
+github.com/containerd/continuity v0.4.2 h1:v3y/4Yz5jwnvqPKJJ+7Wf93fyWoCB3F5EclWG023MDM=
github.com/containerd/fifo v0.0.0-20190226154929-a9fb20d87448/go.mod h1:ODA38xgv3Kuk8dQz2ZQXpnv/UZZUHUCL7pnLehbXgQI=
-github.com/containerd/fifo v0.0.0-20200410184934-f15a3290365b/go.mod h1:jPQ2IAeZRCYxpS/Cm1495vGFww6ecHmMk1YJH2Q5ln0=
-github.com/containerd/fifo v0.0.0-20201026212402-0724c46b320c/go.mod h1:jPQ2IAeZRCYxpS/Cm1495vGFww6ecHmMk1YJH2Q5ln0=
-github.com/containerd/fifo v0.0.0-20210316144830-115abcc95a1d/go.mod h1:ocF/ME1SX5b1AOlWi9r677YJmCPSwwWnQ9O123vzpE4=
-github.com/containerd/fifo v1.0.0/go.mod h1:ocF/ME1SX5b1AOlWi9r677YJmCPSwwWnQ9O123vzpE4=
-github.com/containerd/go-cni v1.0.1/go.mod h1:+vUpYxKvAF72G9i1WoDOiPGRtQpqsNW/ZHtSlv++smU=
-github.com/containerd/go-cni v1.0.2/go.mod h1:nrNABBHzu0ZwCug9Ije8hL2xBCYh/pjfMb1aZGrrohk=
github.com/containerd/go-runc v0.0.0-20180907222934-5a6d9f37cfa3/go.mod h1:IV7qH3hrUgRmyYrtgEeGWJfWbgcHL9CSRruz2Vqcph0=
-github.com/containerd/go-runc v0.0.0-20190911050354-e029b79d8cda/go.mod h1:IV7qH3hrUgRmyYrtgEeGWJfWbgcHL9CSRruz2Vqcph0=
-github.com/containerd/go-runc v0.0.0-20200220073739-7016d3ce2328/go.mod h1:PpyHrqVs8FTi9vpyHwPwiNEGaACDxT/N/pLcvMSRA9g=
-github.com/containerd/go-runc v0.0.0-20201020171139-16b287bc67d0/go.mod h1:cNU0ZbCgCQVZK4lgG3P+9tn9/PaJNmoDXPpoJhDR+Ok=
-github.com/containerd/go-runc v1.0.0/go.mod h1:cNU0ZbCgCQVZK4lgG3P+9tn9/PaJNmoDXPpoJhDR+Ok=
-github.com/containerd/imgcrypt v1.0.1/go.mod h1:mdd8cEPW7TPgNG4FpuP3sGBiQ7Yi/zak9TYCG3juvb0=
-github.com/containerd/imgcrypt v1.0.4-0.20210301171431-0ae5c75f59ba/go.mod h1:6TNsg0ctmizkrOgXRNQjAPFWpMYRWuiB6dSF4Pfa5SA=
-github.com/containerd/imgcrypt v1.1.1-0.20210312161619-7ed62a527887/go.mod h1:5AZJNI6sLHJljKuI9IHnw1pWqo/F0nGDOuR9zgTs7ow=
-github.com/containerd/imgcrypt v1.1.1/go.mod h1:xpLnwiQmEUJPvQoAapeb2SNCxz7Xr6PJrXQb0Dpc4ms=
-github.com/containerd/nri v0.0.0-20201007170849-eb1350a75164/go.mod h1:+2wGSDGFYfE5+So4M5syatU0N0f0LbWpuqyMi4/BE8c=
-github.com/containerd/nri v0.0.0-20210316161719-dbaa18c31c14/go.mod h1:lmxnXF6oMkbqs39FiCt1s0R2HSMhcLel9vNL3m4AaeY=
-github.com/containerd/nri v0.1.0/go.mod h1:lmxnXF6oMkbqs39FiCt1s0R2HSMhcLel9vNL3m4AaeY=
-github.com/containerd/stargz-snapshotter/estargz v0.4.1/go.mod h1:x7Q9dg9QYb4+ELgxmo4gBUeJB0tl5dqH1Sdz0nJU1QM=
github.com/containerd/stargz-snapshotter/estargz v0.14.3 h1:OqlDCK3ZVUO6C3B/5FSkDwbkEETK84kQgEeFwDC+62k=
github.com/containerd/stargz-snapshotter/estargz v0.14.3/go.mod h1:KY//uOCIkSuNAHhJogcZtrNHdKrA99/FCCRjE3HD36o=
github.com/containerd/ttrpc v0.0.0-20190828154514-0e0f228740de/go.mod h1:PvCDdDGpgqzQIzDW1TphrGLssLDZp2GuS+X5DkEJB8o=
-github.com/containerd/ttrpc v0.0.0-20190828172938-92c8520ef9f8/go.mod h1:PvCDdDGpgqzQIzDW1TphrGLssLDZp2GuS+X5DkEJB8o=
-github.com/containerd/ttrpc v0.0.0-20191028202541-4f1b8fe65a5c/go.mod h1:LPm1u0xBw8r8NOKoOdNMeVHSawSsltak+Ihv+etqsE8=
-github.com/containerd/ttrpc v1.0.1/go.mod h1:UAxOpgT9ziI0gJrmKvgcZivgxOp8iFPSk8httJEt98Y=
-github.com/containerd/ttrpc v1.0.2/go.mod h1:UAxOpgT9ziI0gJrmKvgcZivgxOp8iFPSk8httJEt98Y=
-github.com/containerd/ttrpc v1.1.0/go.mod h1:XX4ZTnoOId4HklF4edwc4DcqskFZuvXB1Evzy5KFQpQ=
github.com/containerd/typeurl v0.0.0-20180627222232-a93fcdb778cd/go.mod h1:Cm3kwCdlkCfMSHURc+r6fwoGH6/F1hH3S4sg0rLFWPc=
-github.com/containerd/typeurl v0.0.0-20190911142611-5eb25027c9fd/go.mod h1:GeKYzf2pQcqv7tJ0AoCuuhtnqhva5LNU3U+OyKxxJpk=
-github.com/containerd/typeurl v1.0.1/go.mod h1:TB1hUtrpaiO88KEK56ijojHS1+NeF0izUACaJW2mdXg=
-github.com/containerd/typeurl v1.0.2/go.mod h1:9trJWW2sRlGub4wZJRTW83VtbOLS6hwcDZXTn6oPz9s=
-github.com/containerd/zfs v0.0.0-20200918131355-0a33824f23a2/go.mod h1:8IgZOBdv8fAgXddBT4dBXJPtxyRsejFIpXoklgxgEjw=
-github.com/containerd/zfs v0.0.0-20210301145711-11e8f1707f62/go.mod h1:A9zfAbMlQwE+/is6hi0Xw8ktpL+6glmqZYtevJgaB8Y=
-github.com/containerd/zfs v0.0.0-20210315114300-dde8f0fda960/go.mod h1:m+m51S1DvAP6r3FcmYCp54bQ34pyOwTieQDNRIRHsFY=
-github.com/containerd/zfs v0.0.0-20210324211415-d5c4544f0433/go.mod h1:m+m51S1DvAP6r3FcmYCp54bQ34pyOwTieQDNRIRHsFY=
-github.com/containerd/zfs v1.0.0/go.mod h1:m+m51S1DvAP6r3FcmYCp54bQ34pyOwTieQDNRIRHsFY=
-github.com/containernetworking/cni v0.7.1/go.mod h1:LGwApLUm2FpoOfxTDEeq8T9ipbpZ61X79hmU3w8FmsY=
-github.com/containernetworking/cni v0.8.0/go.mod h1:LGwApLUm2FpoOfxTDEeq8T9ipbpZ61X79hmU3w8FmsY=
-github.com/containernetworking/cni v0.8.1/go.mod h1:LGwApLUm2FpoOfxTDEeq8T9ipbpZ61X79hmU3w8FmsY=
-github.com/containernetworking/plugins v0.8.6/go.mod h1:qnw5mN19D8fIwkqW7oHHYDHVlzhJpcY6TQxn/fUyDDM=
-github.com/containernetworking/plugins v0.9.1/go.mod h1:xP/idU2ldlzN6m4p5LmGiwRDjeJr6FLK6vuiUwoH7P8=
-github.com/containers/common v0.49.1 h1:6y4/s2WwYxrv+Cox7fotOo316wuZI+iKKPUQweCYv50=
-github.com/containers/common v0.49.1/go.mod h1:ueM5hT0itKqCQvVJDs+EtjornAQtrHYxQJzP2gxeGIg=
-github.com/containers/image/v5 v5.24.0 h1:2Pu8ztTntqNxteVN15bORCQnM8rfnbYuyKwUiiKUBuc=
-github.com/containers/image/v5 v5.24.0/go.mod h1:oss5F6ssGQz8ZtC79oY+fuzYA3m3zBek9tq9gmhuvHc=
+github.com/containers/common v0.55.4 h1:7IxB/G5qtDU+rp1YiVWkDpd+ZC4ZlCQ7k2jZJYkB/R8=
+github.com/containers/common v0.55.4/go.mod h1:5mVCpfMBWyO+zaD7Fw+DBHFa42YFKROwle1qpEKcX3U=
+github.com/containers/image/v5 v5.26.2 h1:JX44e1qkdHiL60eooZZcIgKMBp/Ue2AIdKeBG71CYEg=
+github.com/containers/image/v5 v5.26.2/go.mod h1:CTS9VmzgKk8HTSPfPDkthBvRh/2GnFL8K8WBedn3L7I=
github.com/containers/libtrust v0.0.0-20230121012942-c1716e8a8d01 h1:Qzk5C6cYglewc+UyGf6lc8Mj2UaPTHy/iF2De0/77CA=
github.com/containers/libtrust v0.0.0-20230121012942-c1716e8a8d01/go.mod h1:9rfv8iPl1ZP7aqh9YA68wnZv2NUDbXdcdPHVz0pFbPY=
-github.com/containers/ocicrypt v1.0.1/go.mod h1:MeJDzk1RJHv89LjsH0Sp5KTY3ZYkjXO/C+bKAeWFIrc=
-github.com/containers/ocicrypt v1.1.0/go.mod h1:b8AOe0YR67uU8OqfVNcznfFpAzu3rdgUV4GP9qXPfu4=
-github.com/containers/ocicrypt v1.1.1/go.mod h1:Dm55fwWm1YZAjYRaJ94z2mfZikIyIN4B0oB3dj3jFxY=
github.com/containers/ocicrypt v1.1.7 h1:thhNr4fu2ltyGz8aMx8u48Ae0Pnbip3ePP9/mzkZ/3U=
github.com/containers/ocicrypt v1.1.7/go.mod h1:7CAhjcj2H8AYp5YvEie7oVSK2AhBY8NscCYRawuDNtw=
-github.com/containers/storage v1.45.3 h1:GbtTvTtp3GW2/tcFg5VhgHXcYMwVn2KfZKiHjf9FAOM=
-github.com/containers/storage v1.45.3/go.mod h1:OdRUYHrq1HP6iAo79VxqtYuJzC5j4eA2I60jKOoCT7g=
+github.com/containers/storage v1.48.1 h1:mMdr6whnMu8jJ1dO+tKaeSNbu6XJYSufWQF20uLr9Og=
+github.com/containers/storage v1.48.1/go.mod h1:pRp3lkRo2qodb/ltpnudoXggrviRmaCmU5a5GhTBae0=
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.13+incompatible/go.mod h1:uF7uidLiAD3TWHmW31ZFd/JWoc32PjwdhPthX9715RE=
-github.com/coreos/go-iptables v0.4.5/go.mod h1:/mVI274lEDI2ns62jHCDnCyBF9Iwsmekav8Dbxlm1MU=
-github.com/coreos/go-iptables v0.5.0/go.mod h1:/mVI274lEDI2ns62jHCDnCyBF9Iwsmekav8Dbxlm1MU=
-github.com/coreos/go-oidc v2.1.0+incompatible/go.mod h1:CgnwVTmzoESiwO9qyAFEMiHoZ1nMCKZlZ9V6mm3/LKc=
github.com/coreos/go-semver v0.2.0/go.mod h1:nnelYz7RCh+5ahJtPPxZlU+153eP4D4r3EedlOD2RNk=
-github.com/coreos/go-semver v0.3.0 h1:wkHLiw0WNATZnSG7epLsujiMCgPAc9xhjJ4tgnAxmfM=
github.com/coreos/go-semver v0.3.0/go.mod h1:nnelYz7RCh+5ahJtPPxZlU+153eP4D4r3EedlOD2RNk=
-github.com/coreos/go-systemd v0.0.0-20161114122254-48702e0da86b/go.mod h1:F5haX7vjVVG0kc13fIWeqUViNPyEJxv/OmvnBo0Yme4=
-github.com/coreos/go-systemd v0.0.0-20180511133405-39ca1b05acc7/go.mod h1:F5haX7vjVVG0kc13fIWeqUViNPyEJxv/OmvnBo0Yme4=
+github.com/coreos/go-semver v0.3.1 h1:yi21YpKnrx1gt5R+la8n5WgS0kCrsPp33dmEyHReZr4=
+github.com/coreos/go-semver v0.3.1/go.mod h1:irMmmIw/7yzSRPWryHsK7EYSg09caPQL03VsM8rvUec=
github.com/coreos/go-systemd v0.0.0-20190321100706-95778dfbb74e/go.mod h1:F5haX7vjVVG0kc13fIWeqUViNPyEJxv/OmvnBo0Yme4=
github.com/coreos/go-systemd/v22 v22.0.0/go.mod h1:xO0FLkIi5MaZafQlIrOotqXZ90ih+1atmu1JpKERPPk=
-github.com/coreos/go-systemd/v22 v22.1.0/go.mod h1:xO0FLkIi5MaZafQlIrOotqXZ90ih+1atmu1JpKERPPk=
-github.com/coreos/go-systemd/v22 v22.3.2 h1:D9/bQk5vlXQFZ6Kwuu6zaiXJ9oTPe68++AzAJc1DzSI=
github.com/coreos/go-systemd/v22 v22.3.2/go.mod h1:Y58oyj3AT4RCenI/lSvhwexgC+NSVTIJ3seZv2GcEnc=
-github.com/coreos/pkg v0.0.0-20160727233714-3ac0863d7acf/go.mod h1:E3G3o1h8I7cfcXa63jLwjI0eiQQMgzzUDFVpN/nH/eA=
+github.com/coreos/go-systemd/v22 v22.5.0 h1:RrqgGjYQKalulkV8NGVIfkXQf6YYmOyiJKk8iXXhfZs=
+github.com/coreos/go-systemd/v22 v22.5.0/go.mod h1:Y58oyj3AT4RCenI/lSvhwexgC+NSVTIJ3seZv2GcEnc=
github.com/coreos/pkg v0.0.0-20180928190104-399ea9e2e55f/go.mod h1:E3G3o1h8I7cfcXa63jLwjI0eiQQMgzzUDFVpN/nH/eA=
github.com/corpix/uarand v0.1.1 h1:RMr1TWc9F4n5jiPDzFHtmaUXLKLNUFK0SgCLo4BhX/U=
github.com/cpuguy83/go-md2man/v2 v2.0.0-20190314233015-f79a8a8ca69d/go.mod h1:maD7wRr/U5Z6m/iR4s+kqSMx2CaBsrgA7czyZG/E6dU=
github.com/cpuguy83/go-md2man/v2 v2.0.0/go.mod h1:maD7wRr/U5Z6m/iR4s+kqSMx2CaBsrgA7czyZG/E6dU=
github.com/cpuguy83/go-md2man/v2 v2.0.2 h1:p1EgwI/C7NhT0JmVkwCD2ZBK8j4aeHQX2pMHHBfMQ6w=
github.com/cpuguy83/go-md2man/v2 v2.0.2/go.mod h1:tgQtvFlXSQOSOSIRvRPT7W67SCa46tRHOmNcaadrF8o=
-github.com/creack/pty v1.1.7/go.mod h1:lj5s0c3V2DBrqTV7llrYr5NG6My20zk30Fl46Y7DoTY=
github.com/creack/pty v1.1.9/go.mod h1:oKZEueFk5CKHvIhNR5MUki03XCEU+Q6VDXinZuGJ33E=
-github.com/creack/pty v1.1.11/go.mod h1:oKZEueFk5CKHvIhNR5MUki03XCEU+Q6VDXinZuGJ33E=
github.com/creack/pty v1.1.18 h1:n56/Zwd5o6whRC5PMGretI4IdRLlmBXYNjScPaBgsbY=
-github.com/cyberphone/json-canonicalization v0.0.0-20220623050100-57a0ce2678a7 h1:vU+EP9ZuFUCYE0NYLwTSob+3LNEJATzNfP/DC7SWGWI=
-github.com/cyberphone/json-canonicalization v0.0.0-20220623050100-57a0ce2678a7/go.mod h1:uzvlm1mxhHkdfqitSA92i7Se+S9ksOn3a3qmv/kyOCw=
-github.com/cyphar/filepath-securejoin v0.2.2/go.mod h1:FpkQEhXnPnOthhzymB7CGsFk2G9VLXONKD9G7QGMM+4=
-github.com/cyphar/filepath-securejoin v0.2.3 h1:YX6ebbZCZP7VkM3scTTokDgBL2TY741X51MTk3ycuNI=
-github.com/cyphar/filepath-securejoin v0.2.3/go.mod h1:aPGpWjXOXUn2NCNjFvBE6aRxGGx79pTxQpKOJNYHHl4=
-github.com/d2g/dhcp4 v0.0.0-20170904100407-a1d1b6c41b1c/go.mod h1:Ct2BUK8SB0YC1SMSibvLzxjeJLnrYEVLULFNiHY9YfQ=
-github.com/d2g/dhcp4client v1.0.0/go.mod h1:j0hNfjhrt2SxUOw55nL0ATM/z4Yt3t2Kd1mW34z5W5s=
-github.com/d2g/dhcp4server v0.0.0-20181031114812-7d4a0a7f59a5/go.mod h1:Eo87+Kg/IX2hfWJfwxMzLyuSZyxSoAug2nGa1G2QAi8=
-github.com/d2g/hardwareaddr v0.0.0-20190221164911-e7d9fbe030e4/go.mod h1:bMl4RjIciD2oAxI7DmWRx6gbeqrkoLqv3MV0vzNad+I=
-github.com/danieljoos/wincred v1.1.2 h1:QLdCxFs1/Yl4zduvBdcHB8goaYk9RARS2SgLLRuAyr0=
-github.com/danieljoos/wincred v1.1.2/go.mod h1:GijpziifJoIBfYh+S7BbkdUTU4LfM+QnGqR5Vl2tAx0=
+github.com/cyberphone/json-canonicalization v0.0.0-20230514072755-504adb8a8af1 h1:8Pq5UNTC+/UfvcOPKQGZoKCkeF+ZaKa4wJ9OS2gsQQM=
+github.com/cyberphone/json-canonicalization v0.0.0-20230514072755-504adb8a8af1/go.mod h1:uzvlm1mxhHkdfqitSA92i7Se+S9ksOn3a3qmv/kyOCw=
+github.com/cyphar/filepath-securejoin v0.2.4 h1:Ugdm7cg7i6ZK6x3xDF1oEu1nfkyfH53EtKeQYTC3kyg=
+github.com/cyphar/filepath-securejoin v0.2.4/go.mod h1:aPGpWjXOXUn2NCNjFvBE6aRxGGx79pTxQpKOJNYHHl4=
+github.com/danieljoos/wincred v1.2.0 h1:ozqKHaLK0W/ii4KVbbvluM91W2H3Sh0BncbUNPS7jLE=
+github.com/danieljoos/wincred v1.2.0/go.mod h1:FzQLLMKBFdvu+osBrnFODiv32YGwCfx0SkRa/eYHgec=
github.com/dapr/kit v0.11.3 h1:u1X92tE8xsrwXIej7nkcI5Z1t1CFznPwlL18tizNEw4=
github.com/dapr/kit v0.11.3/go.mod h1:hQA6xOhcLAiccXTj7e3/bzpHwvAJCSCp70p2xg3jB40=
github.com/davecgh/go-spew v1.1.0/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38=
@@ -806,21 +670,19 @@ github.com/distribution/distribution/v3 v3.0.0-20221208165359-362910506bc2 h1:aB
github.com/distribution/distribution/v3 v3.0.0-20221208165359-362910506bc2/go.mod h1:WHNsWjnIn2V1LYOrME7e8KxSeKunYHsxEm4am0BUtcI=
github.com/dlclark/regexp2 v1.10.0 h1:+/GIL799phkJqYW+3YbOd8LCcbHzT0Pbo8zl70MHsq0=
github.com/dlclark/regexp2 v1.10.0/go.mod h1:DHkYz0B9wPfa6wondMfaivmHpzrQ3v9q8cnmRbL6yW8=
-github.com/docker/cli v20.10.14+incompatible h1:dSBKJOVesDgHo7rbxlYjYsXe7gPzrTT+/cKQgpDAazg=
-github.com/docker/cli v20.10.14+incompatible/go.mod h1:JLrzqnKDaYBop7H2jaqPtU4hHvMKP+vjCwu2uszcLI8=
+github.com/docker/cli v24.0.6+incompatible h1:fF+XCQCgJjjQNIMjzaSmiKJSCcfcXb3TWTcc7GAneOY=
+github.com/docker/cli v24.0.6+incompatible/go.mod h1:JLrzqnKDaYBop7H2jaqPtU4hHvMKP+vjCwu2uszcLI8=
github.com/docker/distribution v2.8.2+incompatible h1:T3de5rq0dB1j30rp0sA2rER+m322EBzniBPB6ZIzuh8=
github.com/docker/distribution v2.8.2+incompatible/go.mod h1:J2gT2udsDAN96Uj4KfcMRqY0/ypR+oyYUYmja8H+y+w=
github.com/docker/docker-credential-helpers v0.6.3/go.mod h1:WRaJzqw3CTB9bk10avuGsjVBZsD05qeibJ1/TYlvc0Y=
-github.com/docker/docker-credential-helpers v0.7.0 h1:xtCHsjxogADNZcdv1pKUHXryefjlVRqWqIhk/uXJp0A=
-github.com/docker/docker-credential-helpers v0.7.0/go.mod h1:rETQfLdHNT3foU5kuNkFR1R1V12OJRRO5lzt2D1b5X0=
+github.com/docker/docker-credential-helpers v0.8.0 h1:YQFtbBQb4VrpoPxhFuzEBPQ9E16qz5SpHLS+uswaCp8=
+github.com/docker/docker-credential-helpers v0.8.0/go.mod h1:UGFXcuoQ5TxPiB54nHOZ32AWRqQdECoh/Mg0AlEYb40=
github.com/docker/go v1.5.1-1.0.20160303222718-d30aec9fd63c h1:lzqkGL9b3znc+ZUgi7FlLnqjQhcXxkNM/quxIjBVMD0=
github.com/docker/go v1.5.1-1.0.20160303222718-d30aec9fd63c/go.mod h1:CADgU4DSXK5QUlFslkQu2yW2TKzFZcXq/leZfM0UH5Q=
github.com/docker/go-connections v0.4.0/go.mod h1:Gbd7IOopHjR8Iph03tsViu4nIes5XhDvyHbTtUxmeec=
github.com/docker/go-connections v0.4.1-0.20190612165340-fd1b1942c4d5 h1:2o8D0hdBky229bNnc7a8bAZkeVMpH4qsp2Rmt4g/+Zk=
github.com/docker/go-connections v0.4.1-0.20190612165340-fd1b1942c4d5/go.mod h1:Gbd7IOopHjR8Iph03tsViu4nIes5XhDvyHbTtUxmeec=
-github.com/docker/go-events v0.0.0-20170721190031-9461782956ad/go.mod h1:Uw6UezgYA44ePAFQYUehOuCzmy5zmg/+nl2ZfMWGkpA=
github.com/docker/go-events v0.0.0-20190806004212-e31b211e4f1c h1:+pKlWGMw7gf6bQ+oDZB4KHQFypsfjYlq/C4rfL7D3g8=
-github.com/docker/go-events v0.0.0-20190806004212-e31b211e4f1c/go.mod h1:Uw6UezgYA44ePAFQYUehOuCzmy5zmg/+nl2ZfMWGkpA=
github.com/docker/go-metrics v0.0.0-20180209012529-399ea8c73916/go.mod h1:/u0gXw0Gay3ceNrsHubL3BtdOL2fHf93USgMTe0W5dI=
github.com/docker/go-metrics v0.0.1 h1:AgB/0SvBxihN0X8OR4SjsblXkbMvalQ8cjmtKQ2rQV8=
github.com/docker/go-metrics v0.0.1/go.mod h1:cG1hvH2utMXtqgqqYE9plW6lDxS3/5ayHzueweSI3Vw=
@@ -831,14 +693,12 @@ github.com/docker/libtrust v0.0.0-20160708172513-aabc10ec26b7 h1:UhxFibDNY/bfvqU
github.com/docker/libtrust v0.0.0-20160708172513-aabc10ec26b7/go.mod h1:cyGadeNEkKy96OOhEzfZl+yxihPEzKnqJwvfuSUqbZE=
github.com/docker/spdystream v0.0.0-20160310174837-449fdfce4d96/go.mod h1:Qh8CwZgvJUkLughtfhJv5dyTYa91l1fOUCrgjqmcifM=
github.com/docopt/docopt-go v0.0.0-20180111231733-ee0de3bc6815/go.mod h1:WwZ+bS3ebgob9U8Nd0kOddGdZWjyMGR8Wziv+TBNwSE=
-github.com/dustin/go-humanize v0.0.0-20171111073723-bb3d318650d4/go.mod h1:HtrtbFcZ19U5GC7JDqmcUSB87Iq5E25KnS6fMYU6eOk=
github.com/dustin/go-humanize v1.0.0/go.mod h1:HtrtbFcZ19U5GC7JDqmcUSB87Iq5E25KnS6fMYU6eOk=
github.com/dustin/go-humanize v1.0.1 h1:GzkhY7T5VNhEkwH0PVJgjz+fX1rhBrR7pRT3mDkpeCY=
github.com/dustin/go-humanize v1.0.1/go.mod h1:Mu1zIs6XwVuF/gI1OepvI0qD18qycQx+mFykh5fBlto=
github.com/dvsekhvalnov/jose2go v0.0.0-20170216131308-f21a8cedbbae/go.mod h1:7BvyPhdbLxMXIYTFPLsyJRFMsKmOZnQmzh6Gb+uquuM=
github.com/dvsekhvalnov/jose2go v1.5.0 h1:3j8ya4Z4kMCwT5nXIKFSV84YS+HdqSSO0VsTQxaLAeM=
github.com/dvsekhvalnov/jose2go v1.5.0/go.mod h1:QsHjhyTlD/lAVqn/NSbVZmSCGeDehTB/mPZadG+mhXU=
-github.com/dvyukov/go-fuzz v0.0.0-20210103155950-6a8e9d1f2415/go.mod h1:11Gm+ccJnvAhCNLlf5+cS9KjtbaD5I5zaZpFMsTHWTw=
github.com/eapache/go-resiliency v1.3.0 h1:RRL0nge+cWGlxXbUzJ7yMcq6w2XBEr19dCN6HECGaT0=
github.com/eapache/go-resiliency v1.3.0/go.mod h1:5yPzW0MIvSe0JDsv0v+DvcjEv2FyD6iZYSs1ZI+iQho=
github.com/eapache/go-xerial-snappy v0.0.0-20230111030713-bf00bc1b83b6 h1:8yY/I9ndfrgrXUbOGObLHKBR4Fl3nZXwM2c7OYTT8hM=
@@ -846,7 +706,6 @@ github.com/eapache/go-xerial-snappy v0.0.0-20230111030713-bf00bc1b83b6/go.mod h1
github.com/eapache/queue v1.1.0 h1:YOEu7KNc61ntiQlcEeUIoDTJ2o8mQznoNvUhiigpIqc=
github.com/eapache/queue v1.1.0/go.mod h1:6eCeP0CKFpHLu8blIFXhExK/dRa7WDZfr6jVFPTqq+I=
github.com/elazarl/goproxy v0.0.0-20180725130230-947c36da3153/go.mod h1:/Zj4wYkgs4iZTTu3o/KG3Itv/qCCa8VVMlb3i9OVuzc=
-github.com/elazarl/goproxy v0.0.0-20191011121108-aa519ddbe484 h1:pEtiCjIXx3RvGjlUJuCNxNOw0MNblyR9Wi+vJGBFh+8=
github.com/emicklei/go-restful v0.0.0-20170410110728-ff4f55a20633/go.mod h1:otzb+WCGbkyDHkqmQmT5YD2WR4BBwUdeQoFo8l/7tVs=
github.com/emicklei/go-restful v2.9.5+incompatible/go.mod h1:otzb+WCGbkyDHkqmQmT5YD2WR4BBwUdeQoFo8l/7tVs=
github.com/emicklei/go-restful/v3 v3.10.2 h1:hIovbnmBTLjHXkqEBUz3HGpXZdM7ZrE9fJIZIqlJLqE=
@@ -856,17 +715,13 @@ github.com/emicklei/proto v1.10.0/go.mod h1:rn1FgRS/FANiZdD2djyH7TMA9jdRDcYQ9IEN
github.com/emirpasic/gods v1.18.1 h1:FXtiHYKDGKCW2KzwZKx0iC0PQmdlorYgdFG9jPXJ1Bc=
github.com/emirpasic/gods v1.18.1/go.mod h1:8tpGGwCnJ5H4r6BWwaV6OrWmMoPhUl5jm/FMNAnJvWQ=
github.com/envoyproxy/go-control-plane v0.10.2-0.20220325020618-49ff273808a1/go.mod h1:KJwIaB5Mv44NWtYuAOFCVOjcI94vtpEz2JU/D2v6IjE=
-github.com/envoyproxy/go-control-plane v0.10.3 h1:xdCVXxEe0Y3FQith+0cj2irwZudqGYvecuLB1HtdexY=
github.com/envoyproxy/go-control-plane v0.10.3/go.mod h1:fJJn/j26vwOu972OllsvAgJJM//w9BV6Fxbg2LuVd34=
github.com/envoyproxy/protoc-gen-validate v0.1.0/go.mod h1:iSmxcyjqTsJpI2R4NaDN7+kN2VEUnK/pcBlmesArF7c=
github.com/envoyproxy/protoc-gen-validate v0.6.7/go.mod h1:dyJXwwfPK2VSqiB9Klm1J6romD608Ba7Hij42vrOBCo=
-github.com/envoyproxy/protoc-gen-validate v0.9.1 h1:PS7VIOgmSVhWUEeZwTe7z7zouA22Cr590PzXKbZHOVY=
github.com/envoyproxy/protoc-gen-validate v0.9.1/go.mod h1:OKNgG7TCp5pF4d6XftA0++PMirau2/yoOwVac3AbF2w=
github.com/erikstmartin/go-testdb v0.0.0-20160219214506-8d10e4a1bae5/go.mod h1:a2zkGnVExMxdzMo3M0Hi/3sEU+cWnZpSni0O6/Yb/P0=
github.com/estesp/manifest-tool/v2 v2.0.3 h1:F9HMOqcXvtW+8drQB+BjNRU/+bLXOwCfj3mbjqQC2Ns=
github.com/estesp/manifest-tool/v2 v2.0.3/go.mod h1:Suh+tbKQvKHcs4Vltzy8gwZk1y9eSRI635gT4gFw5Ss=
-github.com/etcd-io/gofail v0.0.0-20190801230047-ad7f989257ca/go.mod h1:49H/RkXP8pKaZy4h0d+NW16rSLhyVBt4o6VLJbmOqDE=
-github.com/evanphx/json-patch v0.5.2/go.mod h1:ZWS5hhDbVDyob71nXKNL0+PWn6ToqBHMikGIFbs31qQ=
github.com/evanphx/json-patch v4.9.0+incompatible/go.mod h1:50XU6AFN0ol/bzJsmQLiYLvXMP4fmwYFNcr97nuDLSk=
github.com/evanphx/json-patch v5.6.0+incompatible h1:jBYDEEiFBPxA0v50tFdvOzQQTCvpL6mnFh5mB2/l16U=
github.com/evanphx/json-patch v5.6.0+incompatible/go.mod h1:50XU6AFN0ol/bzJsmQLiYLvXMP4fmwYFNcr97nuDLSk=
@@ -880,34 +735,31 @@ github.com/facebookgo/muster v0.0.0-20150708232844-fd3d7953fd52 h1:a4DFiKFJiDRGF
github.com/fatih/camelcase v1.0.0 h1:hxNvNX/xYBp0ovncs8WyWZrOrpBNub/JfaMvbURyft8=
github.com/fatih/camelcase v1.0.0/go.mod h1:yN2Sb0lFhZJUdVvtELVWefmrXpuZESvPmqwoZc+/fpc=
github.com/fatih/color v1.7.0/go.mod h1:Zm6kSWBoL9eyXnKyktHP6abPY2pDugNf5KwzbycvMj4=
+github.com/fatih/color v1.9.0/go.mod h1:eQcE1qtQxscV5RaZvpXrrb8Drkc3/DdQ+uUYCNjL+zU=
+github.com/fatih/color v1.13.0/go.mod h1:kLAiJbzzSOZDVNGyDpeOxJ47H46qBXwg5ILebYFFOfk=
github.com/fatih/color v1.15.0 h1:kOqh6YHBtK8aywxGerMG2Eq3H6Qgoqeo13Bk2Mv/nBs=
github.com/fatih/color v1.15.0/go.mod h1:0h5ZqXfHYED7Bhv2ZJamyIOUej9KtShiJESRwBDUSsw=
github.com/felixge/httpsnoop v1.0.1/go.mod h1:m8KPJKqk1gH5J9DgRY2ASl2lWCfGKXixSwevea8zH2U=
github.com/felixge/httpsnoop v1.0.3 h1:s/nj+GCswXYzN5v2DpNMuMQYe+0DDwt5WVCU6CWBdXk=
github.com/felixge/httpsnoop v1.0.3/go.mod h1:m8KPJKqk1gH5J9DgRY2ASl2lWCfGKXixSwevea8zH2U=
-github.com/flowstack/go-jsonschema v0.1.1/go.mod h1:yL7fNggx1o8rm9RlgXv7hTBWxdBM0rVwpMwimd3F3N0=
-github.com/form3tech-oss/jwt-go v3.2.2+incompatible/go.mod h1:pbq4aXjuKjdthFRnoDwaVPLA+WlJuPGy+QneDUgJi2k=
github.com/fortytw2/leaktest v1.3.0 h1:u8491cBMTQ8ft8aeV+adlcytMZylmA5nnwwkRZjI8vw=
-github.com/frankban/quicktest v1.11.3/go.mod h1:wRf/ReqHper53s+kmmSZizM8NamnL3IM0I9ntUbOk+k=
+github.com/foxcpp/go-mockdns v1.0.0 h1:7jBqxd3WDWwi/6WhDvacvH1XsN3rOLXyHM1uhvIx6FI=
+github.com/frankban/quicktest v1.14.3/go.mod h1:mgiwOwqx65TmIk1wJ6Q7wvnVMocbUorkibMOrVTHZps=
github.com/frankban/quicktest v1.14.4 h1:g2rn0vABPOOXmZUj+vbmUp0lPoXEMuhTpIluN0XL9UY=
github.com/fsnotify/fsnotify v1.4.7/go.mod h1:jwhsz4b93w/PPRr/qN1Yymfu8t87LnFCMoQvtojpjFo=
github.com/fsnotify/fsnotify v1.4.9/go.mod h1:znqG4EE+3YCdAaPaxE2ZRY/06pZUdp0tY4IgpuI1SZQ=
github.com/fsnotify/fsnotify v1.6.0 h1:n+5WquG0fcWoWp6xPWfHdbskMCQaFnG6PfBrh1Ky4HY=
github.com/fsnotify/fsnotify v1.6.0/go.mod h1:sl3t1tCWJFWoRz9R8WJCbQihKKwmorjAbSClcnxKAGw=
-github.com/fullsailor/pkcs7 v0.0.0-20190404230743-d7302db945fa/go.mod h1:KnogPXtdwXqoenmZCw6S+25EAm2MkxbG0deNDu4cbSA=
-github.com/fullstorydev/grpcurl v1.8.7 h1:xJWosq3BQovQ4QrdPO72OrPiWuGgEsxY8ldYsJbPrqI=
-github.com/fullstorydev/grpcurl v1.8.7/go.mod h1:pVtM4qe3CMoLaIzYS8uvTuDj2jVYmXqMUkZeijnXp/E=
-github.com/fvbommel/sortorder v1.0.2 h1:mV4o8B2hKboCdkJm+a7uX/SIpZob4JzUpc5GGnM45eo=
-github.com/fvbommel/sortorder v1.0.2/go.mod h1:uk88iVf1ovNn1iLfgUVU2F9o5eO30ui720w+kxuqRs0=
+github.com/fvbommel/sortorder v1.1.0 h1:fUmoe+HLsBTctBDoaBwpQo5N+nrCp8g/BjKb/6ZQmYw=
+github.com/fvbommel/sortorder v1.1.0/go.mod h1:uk88iVf1ovNn1iLfgUVU2F9o5eO30ui720w+kxuqRs0=
github.com/garyburd/redigo v1.6.3 h1:HCeeRluvAgMusMomi1+6Y5dmFOdYV/JzoRrrbFlkGIc=
-github.com/getsentry/raven-go v0.2.0/go.mod h1:KungGk8q33+aIAZUIVWZDr2OfAEBsO49PX4NzFV5kcQ=
github.com/ghodss/yaml v0.0.0-20150909031657-73d445a93680/go.mod h1:4dBDuWmgqj2HViK6kFavaiC9ZROes6MMH2rRYeMEF04=
github.com/ghodss/yaml v1.0.0 h1:wQHKEahhL6wmXdzwWG11gIVCkOv05bNOh+Rxn0yngAk=
github.com/ghodss/yaml v1.0.0/go.mod h1:4dBDuWmgqj2HViK6kFavaiC9ZROes6MMH2rRYeMEF04=
github.com/gliderlabs/ssh v0.3.5 h1:OcaySEmAQJgyYcArR+gGGTHCyE7nvhEMTlYY+Dp8CpY=
github.com/gliderlabs/ssh v0.3.5/go.mod h1:8XB4KraRrX39qHhT6yxPsHedjA08I/uBVwj4xC+/+z4=
-github.com/go-errors/errors v1.4.0 h1:2OA7MFw38+e9na72T1xgkomPb6GzZzzxvJ5U630FoRM=
-github.com/go-errors/errors v1.4.0/go.mod h1:sIVyrIiJhuEF+Pj9Ebtd6P/rEYROXFi3BopGUQ5a5Og=
+github.com/go-errors/errors v1.4.2 h1:J6MZopCL4uSllY1OfXM374weqZFFItUbrImctkmUxIA=
+github.com/go-errors/errors v1.4.2/go.mod h1:sIVyrIiJhuEF+Pj9Ebtd6P/rEYROXFi3BopGUQ5a5Og=
github.com/go-git/gcfg v1.5.0 h1:Q5ViNfGF8zFgyJWPqYwA7qGFoMTEiBmdlkcfRmpIMa4=
github.com/go-git/gcfg v1.5.0/go.mod h1:5m20vg6GwYabIxaOonVkTdrILxQMpEShl1xiMF4ua+E=
github.com/go-git/go-billy/v5 v5.3.1/go.mod h1:pmpqyWchKfYfrkb/UVH4otLvyi/5gJlGI4Hb3ZqZ3W0=
@@ -920,10 +772,9 @@ github.com/go-git/go-git/v5 v5.6.1/go.mod h1:mvyoL6Unz0PiTQrGQfSfiLFhBH1c1e84ylC
github.com/go-gl/glfw v0.0.0-20190409004039-e6da0acd62b1/go.mod h1:vR7hzQXu2zJy9AVAgeJqvqgH9Q5CA+iKCZ2gyEVpxRU=
github.com/go-gl/glfw/v3.3/glfw v0.0.0-20191125211704-12ad95a8df72/go.mod h1:tQ2UAYgL5IevRw8kRxooKSPJfGvJ9fJQFa0TUsXzTg8=
github.com/go-gl/glfw/v3.3/glfw v0.0.0-20200222043503-6f7a984d4dc4/go.mod h1:tQ2UAYgL5IevRw8kRxooKSPJfGvJ9fJQFa0TUsXzTg8=
-github.com/go-gorp/gorp/v3 v3.0.2 h1:ULqJXIekoqMx29FI5ekXXFoH1dT2Vc8UhnRzBg+Emz4=
-github.com/go-gorp/gorp/v3 v3.0.2/go.mod h1:BJ3q1ejpV8cVALtcXvXaXyTOlMmJhWDxTmncaR6rwBY=
+github.com/go-gorp/gorp/v3 v3.0.5 h1:PUjzYdYu3HBOh8LE+UUmRG2P0IRDak9XMeGNvaeq4Ow=
+github.com/go-gorp/gorp/v3 v3.0.5/go.mod h1:dLEjIyyRNiXvNZ8PSmzpt1GsWAUK8kjVhEpjH8TixEw=
github.com/go-kit/kit v0.8.0/go.mod h1:xBxKIO96dXMWWy0MnWVtmwkA9/13aqxPnvrjFYMA2as=
-github.com/go-kit/kit v0.9.0/go.mod h1:xBxKIO96dXMWWy0MnWVtmwkA9/13aqxPnvrjFYMA2as=
github.com/go-kit/log v0.1.0/go.mod h1:zbhenjAZHb184qTLMA9ZjW7ThYL0H2mk7Q6pNt4vbaY=
github.com/go-logfmt/logfmt v0.3.0/go.mod h1:Qt1PoO58o5twSAckw1HlFXLmHsOX5/0LbT9GBnD5lWE=
github.com/go-logfmt/logfmt v0.4.0/go.mod h1:3RMwSq7FuexP4Kalkev3ejPJsZTpXXBr9+V4qmtdjCk=
@@ -959,8 +810,8 @@ github.com/go-openapi/jsonreference v0.19.2/go.mod h1:jMjeRr2HHw6nAVajTXJ4eiUwoh
github.com/go-openapi/jsonreference v0.19.3/go.mod h1:rjx6GuL8TTa9VaixXglHmQmIL98+wF9xc8zWvFonSJ8=
github.com/go-openapi/jsonreference v0.19.6/go.mod h1:diGHMEHg2IqXZGKxqyvWdfWU/aim5Dprw5bqpKkTvns=
github.com/go-openapi/jsonreference v0.20.0/go.mod h1:Ag74Ico3lPc+zR+qjn4XBUmXymS4zJbYVCZmcgkasdo=
-github.com/go-openapi/jsonreference v0.20.1 h1:FBLnyygC4/IZZr893oiomc9XaghoveYTrLC1F86HID8=
-github.com/go-openapi/jsonreference v0.20.1/go.mod h1:Bl1zwGIM8/wsvqjsOQLJ/SH+En5Ap4rVB5KVcIDZG2k=
+github.com/go-openapi/jsonreference v0.20.2 h1:3sVjiK66+uXK/6oQ8xgcRKcFgQ5KXa2KvnJRumpMGbE=
+github.com/go-openapi/jsonreference v0.20.2/go.mod h1:Bl1zwGIM8/wsvqjsOQLJ/SH+En5Ap4rVB5KVcIDZG2k=
github.com/go-openapi/loads v0.21.1/go.mod h1:/DtAMXXneXFjbQMGEtbamCZb+4x7eGwkvZCvBmwUG+g=
github.com/go-openapi/loads v0.21.2 h1:r2a/xFIYeZ4Qd2TnGpWDIQNcP80dIaZgf704za8enro=
github.com/go-openapi/loads v0.21.2/go.mod h1:Jq58Os6SSGz0rzh62ptiu8Z31I+OTHqmULx5e/gJbNw=
@@ -982,8 +833,9 @@ github.com/go-openapi/swag v0.19.2/go.mod h1:POnQmlKehdgb5mhVOsnJFsivZCEZ/vjK9gh
github.com/go-openapi/swag v0.19.5/go.mod h1:POnQmlKehdgb5mhVOsnJFsivZCEZ/vjK9gh66Z9tfKk=
github.com/go-openapi/swag v0.19.15/go.mod h1:QYRuS/SOXUCsnplDa677K7+DxSOj6IPNl/eQntq43wQ=
github.com/go-openapi/swag v0.21.1/go.mod h1:QYRuS/SOXUCsnplDa677K7+DxSOj6IPNl/eQntq43wQ=
-github.com/go-openapi/swag v0.22.3 h1:yMBqmnQ0gyZvEb/+KzuWZOXgllrXT4SADYbvDaXHv/g=
github.com/go-openapi/swag v0.22.3/go.mod h1:UzaqsxGiab7freDnrUUra0MwWfN/q7tE4j+VcZ0yl14=
+github.com/go-openapi/swag v0.22.4 h1:QLMzNJnMGPRNDCbySlcj1x01tzU8/9LTTL9hZZZogBU=
+github.com/go-openapi/swag v0.22.4/go.mod h1:UzaqsxGiab7freDnrUUra0MwWfN/q7tE4j+VcZ0yl14=
github.com/go-openapi/validate v0.22.1 h1:G+c2ub6q47kfX1sOBLwIQwzBVt8qmOAARyo/9Fqs9NU=
github.com/go-openapi/validate v0.22.1/go.mod h1:rjnrwK57VJ7A8xqfpAOEKRH8yQSGUriMu5/zuPSQ1hg=
github.com/go-quicktest/qt v1.100.0 h1:I7iSLgIwNp0E0UnSvKJzs7ig0jg/Iq83zsZjtQNW7jY=
@@ -992,8 +844,6 @@ github.com/go-redis/redis/v7 v7.4.1/go.mod h1:JDNMw23GTyLNC4GZu9njt15ctBQVn7xjRf
github.com/go-redis/redismock/v9 v9.0.3 h1:mtHQi2l51lCmXIbTRTqb1EiHYe9tL5Yk5oorlSJJqR0=
github.com/go-redis/redismock/v9 v9.0.3/go.mod h1:F6tJRfnU8R/NZ0E+Gjvoluk14MqMC5ueSZX6vVQypc0=
github.com/go-sql-driver/mysql v1.3.0/go.mod h1:zAC/RDZ24gD3HViQzih4MyKcchzm+sOG5ZlKdlhCg5w=
-github.com/go-sql-driver/mysql v1.4.1/go.mod h1:zAC/RDZ24gD3HViQzih4MyKcchzm+sOG5ZlKdlhCg5w=
-github.com/go-sql-driver/mysql v1.5.0/go.mod h1:DCzpHaOWr8IXmIStZouvnhqoel9Qv2LBy8hT2VhHyBg=
github.com/go-sql-driver/mysql v1.6.0/go.mod h1:DCzpHaOWr8IXmIStZouvnhqoel9Qv2LBy8hT2VhHyBg=
github.com/go-sql-driver/mysql v1.7.1 h1:lUIinVbN1DY0xBg0eMOzmmtGoHwWBbvnWubQUrtU8EI=
github.com/go-sql-driver/mysql v1.7.1/go.mod h1:OXbVy3sEdcQ2Doequ6Z5BW6fXNQTmx+9S1MCJN5yJMI=
@@ -1034,39 +884,29 @@ github.com/gobuffalo/packr/v2 v2.8.3/go.mod h1:0SahksCVcx4IMnigTjiFuyldmTrdTctXs
github.com/gobuffalo/syncx v0.0.0-20190224160051-33c29581e754/go.mod h1:HhnNqWY95UYwwW3uSASeV7vtgYkT2t16hJgV3AEPUpw=
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/godbus/dbus v0.0.0-20151105175453-c7fdd8b5cd55/go.mod h1:/YcGZj5zSblfDWMMoOzV4fas9FZnQYTkDnsGvmh2Grw=
-github.com/godbus/dbus v0.0.0-20180201030542-885f9cc04c9c/go.mod h1:/YcGZj5zSblfDWMMoOzV4fas9FZnQYTkDnsGvmh2Grw=
-github.com/godbus/dbus v0.0.0-20190422162347-ade71ed3457e/go.mod h1:bBOAhwG1umN6/6ZUMtDFBMQR8jRg9O75tm9K00oMsK4=
github.com/godbus/dbus v0.0.0-20190726142602-4481cbc300e2 h1:ZpnhV/YsD2/4cESfV5+Hoeu/iUR3ruzNvZ+yQfO03a0=
github.com/godbus/dbus v0.0.0-20190726142602-4481cbc300e2/go.mod h1:bBOAhwG1umN6/6ZUMtDFBMQR8jRg9O75tm9K00oMsK4=
github.com/godbus/dbus/v5 v5.0.3/go.mod h1:xhWf0FNVPg57R7Z0UbKHbJfkEywrmjJnf7w5xrFpKfA=
github.com/godbus/dbus/v5 v5.0.4/go.mod h1:xhWf0FNVPg57R7Z0UbKHbJfkEywrmjJnf7w5xrFpKfA=
-github.com/godbus/dbus/v5 v5.0.6/go.mod h1:xhWf0FNVPg57R7Z0UbKHbJfkEywrmjJnf7w5xrFpKfA=
github.com/godbus/dbus/v5 v5.1.0 h1:4KLkAxT3aOY8Li4FRJe/KvhoNFFxo0m6fNuFUO8QJUk=
github.com/godbus/dbus/v5 v5.1.0/go.mod h1:xhWf0FNVPg57R7Z0UbKHbJfkEywrmjJnf7w5xrFpKfA=
github.com/godror/godror v0.24.2/go.mod h1:wZv/9vPiUib6tkoDl+AZ/QLf5YZgMravZ7jxH2eQWAE=
github.com/gofrs/flock v0.8.1 h1:+gYjHKf32LDeiEEFhQaotPbLuUXjY5ZqxKgXy7n59aw=
github.com/gofrs/uuid v4.2.0+incompatible h1:yyYWMnhkhrKwwr8gAOcOCYxOOscHgDS9yZgBrnJfGa0=
-github.com/gogo/googleapis v1.2.0/go.mod h1:Njal3psf3qN6dwBtQfUmBZh2ybovJ0tlu3o/AC7HYjU=
-github.com/gogo/googleapis v1.4.0/go.mod h1:5YRNX2z1oM5gXdAkurHa942MDgEJyk02w4OecKY87+c=
github.com/gogo/protobuf v1.0.0/go.mod h1:r8qH/GZQm5c6nD/R0oafs1akxWv10x8SbQlK7atdtwQ=
github.com/gogo/protobuf v1.1.1/go.mod h1:r8qH/GZQm5c6nD/R0oafs1akxWv10x8SbQlK7atdtwQ=
github.com/gogo/protobuf v1.2.1/go.mod h1:hp+jE20tsWTFYpLwKvXlhS1hjn+gTNwPg2I6zVXpSg4=
-github.com/gogo/protobuf v1.2.2-0.20190723190241-65acae22fc9d/go.mod h1:SlYgWuQ5SjCEi6WLHjHCa1yvBfUnHcTbrrZtXPKa29o=
-github.com/gogo/protobuf v1.3.0/go.mod h1:SlYgWuQ5SjCEi6WLHjHCa1yvBfUnHcTbrrZtXPKa29o=
github.com/gogo/protobuf v1.3.1/go.mod h1:SlYgWuQ5SjCEi6WLHjHCa1yvBfUnHcTbrrZtXPKa29o=
github.com/gogo/protobuf v1.3.2 h1:Ov1cvc58UF3b5XjBnZv7+opcTcQFZebYjWzi34vdm4Q=
github.com/gogo/protobuf v1.3.2/go.mod h1:P1XiOD3dCwIKUDQYPy72D8LYyHL2YPYrpS2s69NZV8Q=
github.com/golang-jwt/jwt v3.2.2+incompatible h1:IfV12K8xAKAnZqdXVzCZ+TOjboZ2keLg81eXfW3O+oY=
github.com/golang-jwt/jwt v3.2.2+incompatible/go.mod h1:8pz2t5EyA70fFQQSrl6XZXzqecmYZeUEB8OUGHkxJ+I=
-github.com/golang-jwt/jwt/v4 v4.4.2/go.mod h1:m21LjoU+eqJr34lmDMbreY2eSTRJ1cv77w39/MY0Ch0=
github.com/golang-jwt/jwt/v4 v4.5.0 h1:7cYmW1XlMY7h7ii7UhUyChSgS5wUJEnm9uZVTGqOWzg=
github.com/golang-jwt/jwt/v4 v4.5.0/go.mod h1:m21LjoU+eqJr34lmDMbreY2eSTRJ1cv77w39/MY0Ch0=
github.com/golang-sql/civil v0.0.0-20190719163853-cb61b32ac6fe/go.mod h1:8vg3r2VgvsThLBIFL93Qb5yWzgyZWhEmBwUJWevAkK0=
github.com/golang/glog v0.0.0-20160126235308-23def4e6c14b/go.mod h1:SBH7ygxi8pfUlaOkMMuAQtPIUF8ecWP5IEl/CR7VP2Q=
github.com/golang/glog v1.0.0 h1:nfP3RFugxnNRyKgeWd4oI1nYvXpxrx8ck8ZrcizshdQ=
github.com/golang/glog v1.0.0/go.mod h1:EWib/APOK0SL3dFbYqvxE3UYd8E6s1ouQ7iEp/0LWV4=
-github.com/golang/groupcache v0.0.0-20160516000752-02826c3e7903/go.mod h1:cIg4eruTrX1D+g88fzRXU5OdNfaM+9IcxsU14FzY7Hc=
github.com/golang/groupcache v0.0.0-20190129154638-5b532d6fd5ef/go.mod h1:cIg4eruTrX1D+g88fzRXU5OdNfaM+9IcxsU14FzY7Hc=
github.com/golang/groupcache v0.0.0-20190702054246-869f871628b6/go.mod h1:cIg4eruTrX1D+g88fzRXU5OdNfaM+9IcxsU14FzY7Hc=
github.com/golang/groupcache v0.0.0-20191227052852-215e87163ea7/go.mod h1:cIg4eruTrX1D+g88fzRXU5OdNfaM+9IcxsU14FzY7Hc=
@@ -1116,13 +956,12 @@ github.com/goodhosts/hostsfile v0.1.1 h1:SqRUTFOshOCon0ZSXDrW1bkKZvs4+5pRgYFWySd
github.com/goodhosts/hostsfile v0.1.1/go.mod h1:lXcUP8xO4WR5vvuQ3F/N0bMQoclOtYKEEUnyY2jTusY=
github.com/google/btree v0.0.0-20180813153112-4030bb1f1f0c/go.mod h1:lNA+9X1NB3Zf8V7Ke586lFgjr2dZNuvo3lPJSGZ5JPQ=
github.com/google/btree v1.0.0/go.mod h1:lNA+9X1NB3Zf8V7Ke586lFgjr2dZNuvo3lPJSGZ5JPQ=
-github.com/google/btree v1.0.1/go.mod h1:xXMiIv4Fb/0kKde4SpL7qlzvu5cMJDRkFDxJfI9uaxA=
github.com/google/btree v1.1.2 h1:xf4v41cLI2Z6FxbKm+8Bu+m8ifhj15JuZ9sa0jZCMUU=
github.com/google/btree v1.1.2/go.mod h1:qOPhT0dTNdNzV6Z/lhRX0YXUafgPLFUh+gZMl761Gm4=
github.com/google/certificate-transparency-go v1.0.10-0.20180222191210-5ab67e519c93/go.mod h1:QeJfpSbVSfYc7RgB3gJFj9cbuQMMchQxrWXz8Ruopmg=
-github.com/google/certificate-transparency-go v1.1.3 h1:WEb38wcTe0EuAvg7USzgklnOjjnlMaahYO3faaqnCn8=
-github.com/google/gnostic v0.6.9 h1:ZK/5VhkoX835RikCHpSUJV9a+S3e1zLh59YnyWeBW+0=
-github.com/google/gnostic v0.6.9/go.mod h1:Nm8234We1lq6iB9OmlgNv3nH91XLLVZHCDayfA3xq+E=
+github.com/google/certificate-transparency-go v1.1.5 h1:EVfYyOiMSdwwXd6FJxnh0jYgYj/Dh5n9sXtgIr5+Vj0=
+github.com/google/gnostic-models v0.6.8 h1:yo/ABAfM5IMRsS1VnXjTBvUb61tFIHozhlYvRgGre9I=
+github.com/google/gnostic-models v0.6.8/go.mod h1:5n7qKqH0f5wFt+aWF8CW6pZLLNOfYuF5OpfBSENuI8U=
github.com/google/go-cmp v0.2.0/go.mod h1:oXzfMopK8JAjlY9xF4vHSVASa0yLyX7SntLO5aqRK0M=
github.com/google/go-cmp v0.3.0/go.mod h1:8QqcDgzrUqlUb/G2PQTWiueGozuR1884gddMywk6iLU=
github.com/google/go-cmp v0.3.1/go.mod h1:8QqcDgzrUqlUb/G2PQTWiueGozuR1884gddMywk6iLU=
@@ -1139,9 +978,8 @@ github.com/google/go-cmp v0.5.7/go.mod h1:n+brtR0CgQNWTVd5ZUFpTBC8YFBDLK/h/bpaJ8
github.com/google/go-cmp v0.5.8/go.mod h1:17dUlkBOakJ0+DkrSSNjCkIjxS6bF9zb3elmeNGIjoY=
github.com/google/go-cmp v0.5.9 h1:O2Tfq5qg4qc4AmwVlvv0oLiVAGB7enBSJ2x2DqQFi38=
github.com/google/go-cmp v0.5.9/go.mod h1:17dUlkBOakJ0+DkrSSNjCkIjxS6bF9zb3elmeNGIjoY=
-github.com/google/go-containerregistry v0.5.1/go.mod h1:Ct15B4yir3PLOP5jsy0GNeYVaIZs/MK/Jz5any1wFW0=
-github.com/google/go-containerregistry v0.14.0 h1:z58vMqHxuwvAsVwvKEkmVBz2TlgBgH5k6koEXBtlYkw=
-github.com/google/go-containerregistry v0.14.0/go.mod h1:aiJ2fp/SXvkWgmYHioXnbMdlgB8eXiiYOY55gfN91Wk=
+github.com/google/go-containerregistry v0.16.1 h1:rUEt426sR6nyrL3gt+18ibRcvYpKYdpsa5ZW7MA08dQ=
+github.com/google/go-containerregistry v0.16.1/go.mod h1:u0qB2l7mvtWVR5kNcbFIhFY1hLbf8eeGapA+vbFDCtQ=
github.com/google/go-intervals v0.0.2 h1:FGrVEiUnTRKR8yE04qzXYaJMtnIYqobR5QbblK3ixcM=
github.com/google/go-intervals v0.0.2/go.mod h1:MkaR3LNRfeKLPmqgJYs4E66z5InYjmCjbbr4TQlcT6Y=
github.com/google/gofuzz v1.0.0/go.mod h1:dBl0BpW6vV/+mYPU4Po3pmUjxk6FQPldtuIdl/M65Eg=
@@ -1168,15 +1006,13 @@ github.com/google/pprof v0.0.0-20210226084205-cbba55b83ad5/go.mod h1:kpwsk12EmLe
github.com/google/pprof v0.0.0-20210601050228-01bbb1931b22/go.mod h1:kpwsk12EmLew5upagYY7GY0pfYCcupk39gWOCRROcvE=
github.com/google/pprof v0.0.0-20210609004039-a478d1d731e9/go.mod h1:kpwsk12EmLew5upagYY7GY0pfYCcupk39gWOCRROcvE=
github.com/google/pprof v0.0.0-20210720184732-4bb14d4b1be1/go.mod h1:kpwsk12EmLew5upagYY7GY0pfYCcupk39gWOCRROcvE=
-github.com/google/pprof v0.0.0-20221103000818-d260c55eee4c h1:lvddKcYTQ545ADhBujtIJmqQrZBDsGo7XIMbAQe/sNY=
-github.com/google/pprof v0.0.0-20221103000818-d260c55eee4c/go.mod h1:dDKJzRmX4S37WGHujM7tX//fmj1uioxKzKxz3lo4HJo=
+github.com/google/pprof v0.0.0-20230323073829-e72429f035bd h1:r8yyd+DJDmsUhGrRBxH5Pj7KeFK5l+Y3FsgT8keqKtk=
+github.com/google/pprof v0.0.0-20230323073829-e72429f035bd/go.mod h1:79YE0hCXdHag9sBkw2o+N/YnZtTkXi0UT9Nnixa5eYk=
github.com/google/renameio v0.1.0/go.mod h1:KWCgfxg9yswjAJkECMjeO8J8rahYeXnNhOm40UhjYkI=
-github.com/google/s2a-go v0.1.3 h1:FAgZmpLl/SXurPEZyCMPBIiiYeTbqfjlbdnCNTAkbGE=
-github.com/google/s2a-go v0.1.3/go.mod h1:Ej+mSEMGRnqRzjc7VtF+jdBwYG5fuJfiZ8ELkjEwM0A=
+github.com/google/s2a-go v0.1.4 h1:1kZ/sQM3srePvKs3tXAvQzo66XfcReoqFpIpIccE7Oc=
+github.com/google/s2a-go v0.1.4/go.mod h1:Ej+mSEMGRnqRzjc7VtF+jdBwYG5fuJfiZ8ELkjEwM0A=
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/trillian v1.5.2 h1:roGP6G8aaAch7vP08+oitPkvmZzxjTfIkguozqJ04Ok=
-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.2/go.mod h1:TIyPZe4MgqvfeYDBFedMoGGpEw/LqOeaOT+nhxU+yHo=
github.com/google/uuid v1.2.0/go.mod h1:TIyPZe4MgqvfeYDBFedMoGGpEw/LqOeaOT+nhxU+yHo=
@@ -1197,8 +1033,8 @@ github.com/googleapis/gax-go/v2 v2.4.0/go.mod h1:XOTVJ59hdnfJLIP/dh8n5CGryZR2LxK
github.com/googleapis/gax-go/v2 v2.5.1/go.mod h1:h6B0KMMFNtI2ddbGJn3T3ZbwkeT6yqEF02fYlzkUCyo=
github.com/googleapis/gax-go/v2 v2.6.0/go.mod h1:1mjbznJAPHFpesgE5ucqfYEscaz5kMdcIDwU/6+DDoY=
github.com/googleapis/gax-go/v2 v2.7.0/go.mod h1:TEop28CZZQ2y+c0VxMUmu1lV+fQx57QpBWsYpwqHJx8=
-github.com/googleapis/gax-go/v2 v2.8.0 h1:UBtEZqx1bjXtOQ5BVTkuYghXrr3N4V123VKJK67vJZc=
-github.com/googleapis/gax-go/v2 v2.8.0/go.mod h1:4orTrqY6hXxxaUL4LHIPl6lGo8vAE38/qKbhSAKP6QI=
+github.com/googleapis/gax-go/v2 v2.9.1 h1:DpTpJqzZ3NvX9zqjhIuI1oVzYZMvboZe+3LoeEIJjHM=
+github.com/googleapis/gax-go/v2 v2.9.1/go.mod h1:4FG3gMrVZlyMp5itSYKMU9z/lBE7+SbnUOvzH2HqbEY=
github.com/googleapis/gnostic v0.4.1/go.mod h1:LRhVm6pbyptWbWbuZ38d1eyptfvIytN3ir6b65WBswg=
github.com/googleapis/go-type-adapters v1.0.0/go.mod h1:zHW75FOG2aur7gAO2B+MLby+cLsWGBF62rFAi7WjWO4=
github.com/gopherjs/gopherjs v0.0.0-20181017120253-0766667cb4d1 h1:EGx4pi6eqNxGaHF6qqu48+N2wcFQ5qg5FXgOdqsJ5d8=
@@ -1206,12 +1042,10 @@ github.com/gopherjs/gopherjs v0.0.0-20181017120253-0766667cb4d1/go.mod h1:wJfORR
github.com/gorilla/handlers v1.5.1 h1:9lRY6j8DEeeBT10CvO9hGW0gmky0BprnvDI5vfhUHH4=
github.com/gorilla/handlers v1.5.1/go.mod h1:t8XrUpc4KVXb7HGyJ4/cEnwQiaxrX/hz1Zv/4g96P1Q=
github.com/gorilla/mux v1.7.0/go.mod h1:1lud6UwP+6orDFRuTfBEV8e9/aOM/c4fVVCaMa2zaAs=
-github.com/gorilla/mux v1.7.3/go.mod h1:1lud6UwP+6orDFRuTfBEV8e9/aOM/c4fVVCaMa2zaAs=
github.com/gorilla/mux v1.8.0 h1:i40aqfkR1h2SlN9hojwV5ZA91wcXFOvkdNIeFDP5koI=
github.com/gorilla/mux v1.8.0/go.mod h1:DVbg23sWSpFRCP0SfiEN6jmj59UnW/n46BH5rLB71So=
github.com/gorilla/securecookie v1.1.1/go.mod h1:ra0sb63/xPlUeL+yeDciTfxMRAA+MP+HVt/4epWDjd4=
github.com/gorilla/sessions v1.2.1/go.mod h1:dk2InVEVJ0sfLlnXv9EAgkf6ecYs/i80K/zI+bUmuGM=
-github.com/gorilla/websocket v0.0.0-20170926233335-4201258b820c/go.mod h1:E7qHFY5m1UJ88s3WnNqhKjPHQ0heANvMoAMk2YaljkQ=
github.com/gorilla/websocket v1.4.0/go.mod h1:E7qHFY5m1UJ88s3WnNqhKjPHQ0heANvMoAMk2YaljkQ=
github.com/gorilla/websocket v1.4.2/go.mod h1:YR8l580nyteQvAITg2hZ9XVh4b55+EU/adAjf1fMHhE=
github.com/gorilla/websocket v1.5.0 h1:PPwGk2jz7EePpoHN/+ClbZu8SPxiqlu12wZP/3sWmnc=
@@ -1221,26 +1055,24 @@ github.com/gosuri/uitable v0.0.4/go.mod h1:tKR86bXuXPZazfOTG1FIzvjIdXzd0mo4Vtn16
github.com/gregjones/httpcache v0.0.0-20180305231024-9cad4c3443a7 h1:pdN6V1QBWetyv/0+wjACpqVH+eVULgEjkurDLq3goeM=
github.com/gregjones/httpcache v0.0.0-20180305231024-9cad4c3443a7/go.mod h1:FecbI9+v66THATjSRHfNgh1IVFe/9kFxbXtjV0ctIMA=
github.com/grpc-ecosystem/go-grpc-middleware v1.0.0/go.mod h1:FiyG127CGDf3tlThmgyCl78X/SZQqEOJBCDaAfeWzPs=
-github.com/grpc-ecosystem/go-grpc-middleware v1.0.1-0.20190118093823-f849b5445de4/go.mod h1:FiyG127CGDf3tlThmgyCl78X/SZQqEOJBCDaAfeWzPs=
-github.com/grpc-ecosystem/go-grpc-middleware v1.3.0/go.mod h1:z0ButlSOZa5vEBq9m2m2hlwIgKw+rp3sdCBRoJY+30Y=
github.com/grpc-ecosystem/go-grpc-middleware v1.4.0 h1:UH//fgunKIs4JdUbpDl1VZCDaL56wXCB/5+wF6uHfaI=
github.com/grpc-ecosystem/go-grpc-middleware v1.4.0/go.mod h1:g5qyo/la0ALbONm6Vbp88Yd8NsDy6rZz+RcrMPxvld8=
-github.com/grpc-ecosystem/go-grpc-prometheus v1.2.0 h1:Ovs26xHkKqVztRpIrF/92BcuyuQ/YW4NSIpoGtfXNho=
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.1-0.20210315223345-82c243799c99 h1:JYghRBlGCZyCF2wNUJ8W0cwaQdtpcssJ4CgC406g+WU=
+github.com/grpc-ecosystem/go-grpc-prometheus v1.2.1-0.20210315223345-82c243799c99/go.mod h1:3bDW6wMZJB7tiONtC/1Xpicra6Wp5GgbTbQWCbI5fkc=
github.com/grpc-ecosystem/grpc-gateway v1.9.0/go.mod h1:vNeuVxBJEsws4ogUvrchl83t/GYV9WGTSLVdBhOQFDY=
-github.com/grpc-ecosystem/grpc-gateway v1.9.5/go.mod h1:vNeuVxBJEsws4ogUvrchl83t/GYV9WGTSLVdBhOQFDY=
github.com/grpc-ecosystem/grpc-gateway v1.16.0 h1:gmcG1KaJ57LophUzW0Hy8NmPhnMZb4M0+kPpLofRdBo=
github.com/grpc-ecosystem/grpc-gateway v1.16.0/go.mod h1:BDjrQk3hbvj6Nolgz8mAMFbcEtjT1g+wF4CSlocrBnw=
github.com/grpc-ecosystem/grpc-gateway/v2 v2.7.0/go.mod h1:hgWBS7lorOAVIJEQMi4ZsPv9hVvWI6+ch50m39Pf2Ks=
-github.com/grpc-ecosystem/grpc-gateway/v2 v2.11.3 h1:lLT7ZLSzGLI08vc9cpd+tYmNWjdKDqyr/2L+f6U12Fk=
github.com/grpc-ecosystem/grpc-gateway/v2 v2.11.3/go.mod h1:o//XUCC/F+yRGJoPO/VU0GSB0f8Nhgmxx0VIRUvaC0w=
+github.com/grpc-ecosystem/grpc-gateway/v2 v2.15.2 h1:gDLXvp5S9izjldquuoAhDzccbskOL6tDC5jMSyx3zxE=
+github.com/grpc-ecosystem/grpc-gateway/v2 v2.15.2/go.mod h1:7pdNwVWBBHGiCxa9lAszqCJMbfTISJ7oMftp8+UGV08=
github.com/gsterjov/go-libsecret v0.0.0-20161001094733-a6f4afe4910c h1:6rhixN/i8ZofjG1Y75iExal34USq5p+wiN1tpie8IrU=
github.com/gsterjov/go-libsecret v0.0.0-20161001094733-a6f4afe4910c/go.mod h1:NMPJylDgVpX0MLRlPy15sqSwOFv/U1GZ2m21JhFfek0=
github.com/hailocab/go-hostpool v0.0.0-20160125115350-e80d13ce29ed h1:5upAirOpQc1Q53c0bnx2ufif5kANL7bfZWcc6VJWJd8=
github.com/hailocab/go-hostpool v0.0.0-20160125115350-e80d13ce29ed/go.mod h1:tMWxXQ9wFIaZeTI9F+hmhFiGpFmhOHzyShyFUhRm0H4=
github.com/hashicorp/consul/api v1.1.0/go.mod h1:VmuI/Lkw1nC05EYQWNKwWGbkg+FbDBtguAZLlVdkD9Q=
github.com/hashicorp/consul/sdk v0.1.1/go.mod h1:VKf9jXwCTEY1QZP2MOLRhb5i/I/ssyNV1vwHyQBF0x8=
-github.com/hashicorp/errwrap v0.0.0-20141028054710-7554cd9344ce/go.mod h1:YH+1FKiLXxHSkmPseP+kNlulaMuP3n2brvKWEqk/Jc4=
github.com/hashicorp/errwrap v1.0.0/go.mod h1:YH+1FKiLXxHSkmPseP+kNlulaMuP3n2brvKWEqk/Jc4=
github.com/hashicorp/errwrap v1.1.0 h1:OxrOeh75EUXMY8TBjag2fzXGZ40LB6IKw45YeGUDY2I=
github.com/hashicorp/errwrap v1.1.0/go.mod h1:YH+1FKiLXxHSkmPseP+kNlulaMuP3n2brvKWEqk/Jc4=
@@ -1249,14 +1081,13 @@ github.com/hashicorp/go-cleanhttp v0.5.2 h1:035FKYIWjmULyFRBKPs8TBQoi0x6d9G4xc9n
github.com/hashicorp/go-cleanhttp v0.5.2/go.mod h1:kO/YDlP8L1346E6Sodw+PrpBSV4/SoxCXGY6BqNFT48=
github.com/hashicorp/go-getter v1.7.0 h1:bzrYP+qu/gMrL1au7/aDvkoOVGUJpeKBgbqRHACAFDY=
github.com/hashicorp/go-getter v1.7.0/go.mod h1:W7TalhMmbPmsSMdNjD0ZskARur/9GJ17cfHTRtXV744=
-github.com/hashicorp/go-hclog v1.3.1 h1:vDwF1DFNZhntP4DAjuTpOw3uEgMUpXh1pB5fW9DqHpo=
+github.com/hashicorp/go-hclog v1.2.0 h1:La19f8d7WIlm4ogzNHB0JGqs5AUDAZ2UfCY4sJXcJdM=
github.com/hashicorp/go-immutable-radix v1.0.0/go.mod h1:0y9vanUI8NX6FsYoO3zeMjhV/C5i9g4Q3DwcSNZ4P60=
github.com/hashicorp/go-msgpack v0.5.3/go.mod h1:ahLV/dePpqEmjfWmKiqvPkv/twdG7iPBM1vqhUKIvfM=
-github.com/hashicorp/go-multierror v0.0.0-20161216184304-ed905158d874/go.mod h1:JMRHfdO9jKNzS/+BTlxCjKNQHg/jZAft8U7LloJvN7I=
github.com/hashicorp/go-multierror v1.0.0/go.mod h1:dHtQlpGsu+cZNNAkkCN/P3hoUDHhCYQXV3UM06sGGrk=
github.com/hashicorp/go-multierror v1.1.1 h1:H5DkEtf6CXdFp0N0Em5UCwQpXMWke8IA0+lD48awMYo=
github.com/hashicorp/go-multierror v1.1.1/go.mod h1:iw975J/qwKPdAO1clOe2L8331t/9/fmwbPZ6JB6eMoM=
-github.com/hashicorp/go-plugin v1.4.5 h1:oTE/oQR4eghggRg8VY7PAz3dr++VwDNBGCcOfIvHpBo=
+github.com/hashicorp/go-plugin v1.4.3 h1:DXmvivbWD5qdiBts9TpBC7BYL1Aia5sxbRgQB+v6UZM=
github.com/hashicorp/go-rootcerts v1.0.0/go.mod h1:K6zTfqpRlCUIjkwsN4Z+hiSfzSTQa6eBIzfwKfwNnHU=
github.com/hashicorp/go-safetemp v1.0.0 h1:2HR189eFNrjHQyENnQMMpCiBAsRxzbTMIgBhEyExpmo=
github.com/hashicorp/go-safetemp v1.0.0/go.mod h1:oaerMy3BhqiTbVye6QuFhFtIceqFoDHxNAB65b+Rj1I=
@@ -1285,29 +1116,27 @@ github.com/hashicorp/terraform-exec v0.18.0 h1:BJa6/Fhxnb0zvsEGqUrFSybcnhAiBVSUg
github.com/hashicorp/terraform-exec v0.18.0/go.mod h1:6PMRgg0Capig5Fn0zW9/+WM3vQsdwotwa8uxDVzLpHE=
github.com/hashicorp/terraform-json v0.15.0 h1:/gIyNtR6SFw6h5yzlbDbACyGvIhKtQi8mTsbkNd79lE=
github.com/hashicorp/terraform-json v0.15.0/go.mod h1:+L1RNzjDU5leLFZkHTFTbJXaoqUC6TqXlFgDoOXrtvk=
-github.com/hashicorp/yamux v0.1.1 h1:yrQxtgseBDrq9Y652vSRDvsKCJKOUD+GzTS4Y0Y8pvE=
+github.com/hashicorp/yamux v0.0.0-20180604194846-3520598351bb h1:b5rjCoWHc7eqmAS4/qyk21ZsHyb6Mxv/jykxvNTkU4M=
github.com/honeycombio/beeline-go v1.10.0 h1:cUDe555oqvw8oD76BQJ8alk7FP0JZ/M/zXpNvOEDLDc=
github.com/honeycombio/libhoney-go v1.16.0 h1:kPpqoz6vbOzgp7jC6SR7SkNj7rua7rgxvznI6M3KdHc=
github.com/hpcloud/tail v1.0.0/go.mod h1:ab1qPbhIpdTxEkNHXyeSf5vhxWSCs/tWer42PpOxQnU=
github.com/huandu/xstrings v1.3.1/go.mod h1:y5/lhBue+AyNmUVz9RLU9xbLR0o4KIIExikq4ovT0aE=
github.com/huandu/xstrings v1.3.2/go.mod h1:y5/lhBue+AyNmUVz9RLU9xbLR0o4KIIExikq4ovT0aE=
-github.com/huandu/xstrings v1.3.3 h1:/Gcsuc1x8JVbJ9/rlye4xZnVAbEkGauT8lbebqcQws4=
github.com/huandu/xstrings v1.3.3/go.mod h1:y5/lhBue+AyNmUVz9RLU9xbLR0o4KIIExikq4ovT0aE=
+github.com/huandu/xstrings v1.4.0 h1:D17IlohoQq4UcpqD7fDk80P7l+lwAmlFaBHgOipl2FU=
+github.com/huandu/xstrings v1.4.0/go.mod h1:y5/lhBue+AyNmUVz9RLU9xbLR0o4KIIExikq4ovT0aE=
github.com/iancoleman/strcase v0.2.0/go.mod h1:iwCmte+B7n89clKwxIoIXy/HfoL7AsD47ZCWhYzw7ho=
github.com/ianlancetaylor/demangle v0.0.0-20181102032728-5e5cf60278f6/go.mod h1:aSSvb/t6k1mPoxDqO4vJh6VOCGPwU4O0C2/Eqndh1Sc=
github.com/ianlancetaylor/demangle v0.0.0-20200824232613-28f6c0f3b639/go.mod h1:aSSvb/t6k1mPoxDqO4vJh6VOCGPwU4O0C2/Eqndh1Sc=
github.com/icrowley/fake v0.0.0-20180203215853-4178557ae428 h1:Mo9W14pwbO9VfRe+ygqZ8dFbPpoIK1HFrG/zjTuQ+nc=
github.com/imdario/mergo v0.3.5/go.mod h1:2EnlNZ0deacrJVfApfmtdGgDfMuh/nq6Ok1EcJh5FfA=
-github.com/imdario/mergo v0.3.8/go.mod h1:2EnlNZ0deacrJVfApfmtdGgDfMuh/nq6Ok1EcJh5FfA=
-github.com/imdario/mergo v0.3.10/go.mod h1:jmQim1M+e3UYxmgPu/WyfjB3N3VflVyUjjjwH0dnCYA=
github.com/imdario/mergo v0.3.11/go.mod h1:jmQim1M+e3UYxmgPu/WyfjB3N3VflVyUjjjwH0dnCYA=
-github.com/imdario/mergo v0.3.12/go.mod h1:jmQim1M+e3UYxmgPu/WyfjB3N3VflVyUjjjwH0dnCYA=
-github.com/imdario/mergo v0.3.13 h1:lFzP57bqS/wsqKssCGmtLAb8A0wKjLGrve2q3PPVcBk=
github.com/imdario/mergo v0.3.13/go.mod h1:4lJ1jqUDcsbIECGy0RUJAXNIhg+6ocWgb1ALK2O4oXg=
+github.com/imdario/mergo v0.3.14 h1:fOqeC1+nCuuk6PKQdg9YmosXX7Y7mHX6R/0ZldI9iHo=
+github.com/imdario/mergo v0.3.14/go.mod h1:WBLT9ZmE3lPoWsEzCh9LPo3TiwVN+ZKEjmz+hD27ysY=
github.com/inconshreveable/mousetrap v1.0.0/go.mod h1:PxqpIevigyE2G7u3NXJIT2ANytuPF1OarO4DADm73n8=
github.com/inconshreveable/mousetrap v1.1.0 h1:wN+x4NVGpMsO7ErUn/mUI3vEoE6Jt13X2s0bqwp9tc8=
github.com/inconshreveable/mousetrap v1.1.0/go.mod h1:vpF70FUmC8bwa3OWnCshd2FqLfsEA9PFc4w1p2J65bw=
-github.com/j-keck/arping v0.0.0-20160618110441-2cf9dc699c56/go.mod h1:ymszkNOg6tORTn+6F6j+Jc8TOr5osrynvN6ivFWZ2GA=
github.com/jackc/pgpassfile v1.0.0 h1:/6Hmqy13Ss2zCq62VdNG8tM1wchn8zjSGOBJ6icpsIM=
github.com/jackc/pgpassfile v1.0.0/go.mod h1:CEx0iS5ambNFdcRtxPj5JhEz+xB6uRky5eyVu/W2HEg=
github.com/jackc/pgservicefile v0.0.0-20221227161230-091c0ba34f0a h1:bbPeKD0xmW/Y25WS6cokEszi5g+S0QxI/d45PkRi7Nk=
@@ -1334,13 +1163,6 @@ github.com/jedib0t/go-pretty/v6 v6.4.6 h1:v6aG9h6Uby3IusSSEjHaZNXpHFhzqMmjXcPq1R
github.com/jedib0t/go-pretty/v6 v6.4.6/go.mod h1:Ndk3ase2CkQbXLLNf5QDHoYb6J9WtVfmHZu9n8rk2xs=
github.com/jessevdk/go-flags v1.4.0/go.mod h1:4FA24M0QyGHXBuZZK/XkWh8h0e1EYbRYJSGM75WSRxI=
github.com/jessevdk/go-flags v1.5.0/go.mod h1:Fw0T6WPc1dYxT4mKEZRfG5kJhaTDP9pj1c2EWnYs/m4=
-github.com/jhump/gopoet v0.0.0-20190322174617-17282ff210b3/go.mod h1:me9yfT6IJSlOL3FCfrg+L6yzUEZ+5jW6WHt4Sk+UPUI=
-github.com/jhump/gopoet v0.1.0/go.mod h1:me9yfT6IJSlOL3FCfrg+L6yzUEZ+5jW6WHt4Sk+UPUI=
-github.com/jhump/goprotoc v0.5.0/go.mod h1:VrbvcYrQOrTi3i0Vf+m+oqQWk9l72mjkJCYo7UvLHRQ=
-github.com/jhump/protoreflect v1.11.0/go.mod h1:U7aMIjN0NWq9swDP7xDdoMfRHb35uiuTd3Z9nFXJf5E=
-github.com/jhump/protoreflect v1.12.0/go.mod h1:JytZfP5d0r8pVNLZvai7U/MCuTWITgrI4tTg7puQFKI=
-github.com/jhump/protoreflect v1.14.1 h1:N88q7JkxTHWFEqReuTsYH1dPIwXxA0ITNQp7avLY10s=
-github.com/jhump/protoreflect v1.14.1/go.mod h1:JytZfP5d0r8pVNLZvai7U/MCuTWITgrI4tTg7puQFKI=
github.com/jinzhu/gorm v0.0.0-20170222002820-5409931a1bb8 h1:CZkYfurY6KGhVtlalI4QwQ6T0Cu6iuY3e0x5RLu96WE=
github.com/jinzhu/gorm v0.0.0-20170222002820-5409931a1bb8/go.mod h1:Vla75njaFJ8clLU1W44h34PjIkijhjHIYnZxMqCdxqo=
github.com/jinzhu/inflection v0.0.0-20170102125226-1c35d901db3d h1:jRQLvyVGL+iVtDElaEIDdKwpPqUIZJfzkNLV34htpEc=
@@ -1353,16 +1175,13 @@ github.com/jmespath/go-jmespath/internal/testify v1.5.1/go.mod h1:L3OGu8Wl2/fWfC
github.com/jmhodges/clock v0.0.0-20160418191101-880ee4c33548 h1:dYTbLf4m0a5u0KLmPfB6mgxbcV7588bOCx79hxa5Sr4=
github.com/jmoiron/sqlx v1.3.5 h1:vFFPA71p1o5gAeqtEAwLU4dnX2napprKtHr7PYIcN3g=
github.com/jmoiron/sqlx v1.3.5/go.mod h1:nRVWtLre0KfCLJvgxzCsLVMogSvQ1zNJtpYr2Ccp0mQ=
-github.com/joefitzgerald/rainbow-reporter v0.1.0/go.mod h1:481CNgqmVHQZzdIbN52CupLJyoVwB10FQ/IQlF1pdL8=
github.com/joho/godotenv v1.3.0 h1:Zjp+RcGpHhGlrMbJzXTrZZPrWj+1vfm90La1wgB6Bhc=
github.com/joho/godotenv v1.3.0/go.mod h1:7hK45KPybAkOC6peb+G5yklZfMxEjkZhHbwpqxOKXbg=
github.com/jonboulle/clockwork v0.1.0/go.mod h1:Ii8DK3G1RaLaWxj9trq07+26W01tbo22gdxWY5EU2bo=
-github.com/jonboulle/clockwork v0.2.2/go.mod h1:Pkfl5aHPm1nk2H9h0bjmnJD/BcgbGXUBGnn1kMkgxc8=
github.com/jonboulle/clockwork v0.3.0 h1:9BSCMi8C+0qdApAp4auwX0RkLGUjs956h0EkuQymUhg=
github.com/jonboulle/clockwork v0.3.0/go.mod h1:Pkfl5aHPm1nk2H9h0bjmnJD/BcgbGXUBGnn1kMkgxc8=
github.com/josharian/intern v1.0.0 h1:vlS4z54oSdjm0bgjRigI+G1HpF+tI+9rE5LLzOg8HmY=
github.com/josharian/intern v1.0.0/go.mod h1:5DoeVV0s6jJacbCEi61lwdGj/aVlrQvzHFFd8Hwg//Y=
-github.com/jpillora/backoff v1.0.0/go.mod h1:J/6gKK9jxlEcS3zixgDgUAsiuZ7yrSoa/FX5e0EB2j4=
github.com/json-iterator/go v1.1.6/go.mod h1:+SdeFBvtyEkXs7REEP0seUULqWtbJapLOCVDaaPEHmU=
github.com/json-iterator/go v1.1.7/go.mod h1:KdQUCv79m/52Kvf8AW2vK1V8akMuk1QjK/uOdHXbAo4=
github.com/json-iterator/go v1.1.10/go.mod h1:KdQUCv79m/52Kvf8AW2vK1V8akMuk1QjK/uOdHXbAo4=
@@ -1375,9 +1194,8 @@ github.com/jtolds/gls v4.20.0+incompatible h1:xdiiI2gbIgH/gLH7ADydsJ1uDOEzR8yvV7
github.com/jtolds/gls v4.20.0+incompatible/go.mod h1:QJZ7F/aHp+rZTRtaJ1ow/lLfFfVYBRgL+9YlvaHOwJU=
github.com/juju/loggo v0.0.0-20190526231331-6e530bcce5d8/go.mod h1:vgyd7OREkbtVEN/8IXZe5Ooef3LQePvuBm9UWj6ZL8U=
github.com/julienschmidt/httprouter v1.2.0/go.mod h1:SYymIcj16QtmaHHD7aYtjjsJG7VTCxuUUipMqKk8s4w=
-github.com/julienschmidt/httprouter v1.3.0/go.mod h1:JR6WtHb+2LUe8TCKY3cZOxFyyO8IZAc4RVcycCCAKdM=
-github.com/k3d-io/k3d/v5 v5.4.4 h1:txQNPNhxuSDRU1+dN9wDjliQp9DslK76gVplqWjQDjg=
-github.com/k3d-io/k3d/v5 v5.4.4/go.mod h1:+eDAoEC4G3bJ4daG2h68kY+L4mCkV7GU4TTGXhfSk0c=
+github.com/k3d-io/k3d/v5 v5.6.0 h1:XMRSQXyPErOcDCdOJVi6HUPjJZuWd/N6Dss7QeCDRhk=
+github.com/k3d-io/k3d/v5 v5.6.0/go.mod h1:t/hRD2heCSkO9TJJdzFT72jXGCY8PjsCsClgjcmMoAA=
github.com/kardianos/osext v0.0.0-20190222173326-2bc1f35cddc0 h1:iQTw/8FWTuc7uiaSepXwyf3o52HaUYcV+Tu66S3F5GA=
github.com/karrick/godirwalk v1.8.0/go.mod h1:H5KPZjojv4lE+QYImBI8xVtrBRgYrIVsaRPx4tDPEn4=
github.com/karrick/godirwalk v1.10.3/go.mod h1:RoGL9dQei4vP9ilrpETWE8CLOZ1kiN0LhBygSwrAsHA=
@@ -1389,15 +1207,13 @@ github.com/kisielk/errcheck v1.1.0/go.mod h1:EZBBE59ingxPouuu3KfxchcWSUPOHkagtvW
github.com/kisielk/errcheck v1.2.0/go.mod h1:/BMXB+zMLi60iA8Vv6Ksmxu/1UDYcXs4uQLJ+jE2L00=
github.com/kisielk/errcheck v1.5.0/go.mod h1:pFxgyoBC7bSaBwPgfKdkLd5X25qrDl4LWUI2bnpBCr8=
github.com/kisielk/gotool v1.0.0/go.mod h1:XhKaO+MFFWcvkIS/tQcRk01m1F5IRFswLeQ+oQHNcck=
-github.com/klauspost/compress v1.11.3/go.mod h1:aoV0uJVorq1K+umq18yTdKaF57EivdYsUV+/s2qKfXs=
-github.com/klauspost/compress v1.11.13/go.mod h1:aoV0uJVorq1K+umq18yTdKaF57EivdYsUV+/s2qKfXs=
github.com/klauspost/compress v1.13.6/go.mod h1:/3/Vjq9QcHkK5uEr5lBEmyoZ1iFhe47etQ6QUkpK6sk=
github.com/klauspost/compress v1.15.11/go.mod h1:QPwzmACJjUTFsnSHH934V6woptycfrDDJnH7hvFVbGM=
-github.com/klauspost/compress v1.16.3 h1:XuJt9zzcnaz6a16/OU53ZjWp/v7/42WcR5t2a0PcNQY=
-github.com/klauspost/compress v1.16.3/go.mod h1:ntbaceVETuRiXiv4DpjP66DpAtAGkEQskQzEyD//IeE=
+github.com/klauspost/compress v1.16.6 h1:91SKEy4K37vkp255cJ8QesJhjyRO0hn9i9G0GoUwLsk=
+github.com/klauspost/compress v1.16.6/go.mod h1:ntbaceVETuRiXiv4DpjP66DpAtAGkEQskQzEyD//IeE=
github.com/klauspost/cpuid/v2 v2.2.4 h1:acbojRNwl3o09bUq+yDCtZFc1aiwaAAxtcn8YkZXnvk=
-github.com/klauspost/pgzip v1.2.6-0.20220930104621-17e8dac29df8 h1:BcxbplxjtczA1a6d3wYoa7a0WL3rq9DKBMGHeKyjEF0=
-github.com/klauspost/pgzip v1.2.6-0.20220930104621-17e8dac29df8/go.mod h1:Ch1tH69qFZu15pkjo5kYi6mth2Zzwzt50oCQKQE9RUs=
+github.com/klauspost/pgzip v1.2.6 h1:8RXeL5crjEUFnR2/Sn6GJNWtSQ3Dk8pq4CL3jvdDyjU=
+github.com/klauspost/pgzip v1.2.6/go.mod h1:Ch1tH69qFZu15pkjo5kYi6mth2Zzwzt50oCQKQE9RUs=
github.com/konsorten/go-windows-terminal-sequences v1.0.1/go.mod h1:T0+1ngSBFLxvqU3pZ+m/2kptfBszLMUkC4ZK/EgS/cQ=
github.com/konsorten/go-windows-terminal-sequences v1.0.2/go.mod h1:T0+1ngSBFLxvqU3pZ+m/2kptfBszLMUkC4ZK/EgS/cQ=
github.com/konsorten/go-windows-terminal-sequences v1.0.3/go.mod h1:T0+1ngSBFLxvqU3pZ+m/2kptfBszLMUkC4ZK/EgS/cQ=
@@ -1410,7 +1226,9 @@ github.com/kr/logfmt v0.0.0-20140226030751-b84e30acd515/go.mod h1:+0opPa2QZZtGFB
github.com/kr/pretty v0.1.0/go.mod h1:dAy3ld7l9f0ibDNOQOHHMYYIIbhfbHSm3C4ZsoJORNo=
github.com/kr/pretty v0.2.0/go.mod h1:ipq/a2n7PKx3OHsz4KJII5eveXtPO4qwEXGdVfWzfnI=
github.com/kr/pretty v0.2.1/go.mod h1:ipq/a2n7PKx3OHsz4KJII5eveXtPO4qwEXGdVfWzfnI=
+github.com/kr/pretty v0.3.0/go.mod h1:640gp4NfQd8pI5XOwp5fnNeVWj67G7CFk/SaSQn7NBk=
github.com/kr/pretty v0.3.1 h1:flRD4NNwYAUpkphVc1HcthR4KEIFJ65n8Mw5qdRn3LE=
+github.com/kr/pretty v0.3.1/go.mod h1:hoEshYVHaxMs3cyo3Yncou5ZscifuDolrwPKZanG3xk=
github.com/kr/pty v1.1.1/go.mod h1:pFQYn66WHrOpPYNljwOMqo10TkYh1fy3cYio2l3bCsQ=
github.com/kr/pty v1.1.5/go.mod h1:9r2w37qlBe7rQ6e1fg1S/9xpWHSnaqNdHD3WcMdbPDA=
github.com/kr/text v0.1.0/go.mod h1:4Jbv+DJW3UT/LiOwJeYQe1efqtUx/iVham/4vfdArNI=
@@ -1440,16 +1258,15 @@ github.com/lestrrat-go/file-rotatelogs v2.4.0+incompatible h1:Y6sqxHMyB1D2YSzWkL
github.com/lestrrat-go/file-rotatelogs v2.4.0+incompatible/go.mod h1:ZQnN8lSECaebrkQytbHj4xNgtg8CR7RYXnPok8e0EHA=
github.com/lestrrat-go/strftime v1.0.5 h1:A7H3tT8DhTz8u65w+JRpiBxM4dINQhUXAZnhBa2xeOE=
github.com/lestrrat-go/strftime v1.0.5/go.mod h1:E1nN3pCbtMSu1yjSVeyuRFVm/U0xoR76fd03sz+Qz4g=
-github.com/letsencrypt/boulder v0.0.0-20221109233200-85aa52084eaf h1:ndns1qx/5dL43g16EQkPV/i8+b3l5bYQwLeoSBe7tS8=
-github.com/letsencrypt/boulder v0.0.0-20221109233200-85aa52084eaf/go.mod h1:aGkAgvWY/IUcVFfuly53REpfv5edu25oij+qHRFaraA=
+github.com/letsencrypt/boulder v0.0.0-20230213213521-fdfea0d469b6 h1:unJdfS94Y3k85TKy+mvKzjW5R9rIC+Lv4KGbE7uNu0I=
+github.com/letsencrypt/boulder v0.0.0-20230213213521-fdfea0d469b6/go.mod h1:PUgW5vI9ANEaV6qv9a6EKu8gAySgwf0xrzG9xIB/CK0=
github.com/lib/pq v0.0.0-20150723085316-0dad96c0b94f/go.mod h1:5WUZQaWbwv1U+lTReE5YruASi9Al49XbQIvNi/34Woo=
github.com/lib/pq v1.2.0/go.mod h1:5WUZQaWbwv1U+lTReE5YruASi9Al49XbQIvNi/34Woo=
-github.com/lib/pq v1.10.0/go.mod h1:AlVN5x4E4T544tWzH6hKfbfQvm3HdbOxrmggDNAPY9o=
+github.com/lib/pq v1.10.7/go.mod h1:AlVN5x4E4T544tWzH6hKfbfQvm3HdbOxrmggDNAPY9o=
github.com/lib/pq v1.10.9 h1:YXG7RB+JIjhP29X+OtkiDnYaXQwpS4JEWq7dtCCRUEw=
github.com/lib/pq v1.10.9/go.mod h1:AlVN5x4E4T544tWzH6hKfbfQvm3HdbOxrmggDNAPY9o=
github.com/liggitt/tabwriter v0.0.0-20181228230101-89fcab3d43de h1:9TO3cAIGXtEhnIaL+V+BEER86oLrvS+kWobKpbJuye0=
github.com/liggitt/tabwriter v0.0.0-20181228230101-89fcab3d43de/go.mod h1:zAbeS9B/r2mtpb6U+EI2rYA5OAXxsYw6wTamcNW+zcE=
-github.com/linuxkit/virtsock v0.0.0-20201010232012-f8cee7dfc7a3/go.mod h1:3r6x7q95whyfWQpmGZTu3gk3v2YkMi05HEzl7Tf7YEo=
github.com/lithammer/dedent v1.1.0 h1:VNzHMVCBNG1j0fh3OrsFRkVUwStdDArbgBWoPAffktY=
github.com/lithammer/dedent v1.1.0/go.mod h1:jrXYCQtgg0nJiN+StA2KgR7w6CiQNv9Fd/Z9BP0jIOc=
github.com/longhorn/go-iscsi-helper v0.0.0-20210330030558-49a327fb024e h1:hz4quJkaJWDo+xW+G6wTF6d6/95QvJ+o2D0+bB/tJ1U=
@@ -1485,12 +1302,19 @@ github.com/matryer/is v1.2.0/go.mod h1:2fLPjFQM9rhQ15aVEtbuwhJinnOqrmgXPNdZsdwlW
github.com/matryer/is v1.4.0 h1:sosSmIWwkYITGrxZ25ULNDeKiMNzFSr4V/eqBQP0PeE=
github.com/matryer/is v1.4.0/go.mod h1:8I/i5uYgLzgsgEloJE1U6xx5HkBQpAZvepWuujKwMRU=
github.com/mattn/go-colorable v0.0.9/go.mod h1:9vuHe8Xs5qXnSaW/c/ABM9alt+Vo+STaOChaDxuIBZU=
+github.com/mattn/go-colorable v0.1.4/go.mod h1:U0ppj6V5qS13XJ6of8GYAs25YV2eR4EVcfRqFIhoBtE=
+github.com/mattn/go-colorable v0.1.9/go.mod h1:u6P/XSegPjTcexA+o6vUJrdnUu04hMope9wVRipJSqc=
github.com/mattn/go-colorable v0.1.13 h1:fFA4WZxdEF4tXPZVKMLwD8oUnCTTo08duU7wxecdEvA=
github.com/mattn/go-colorable v0.1.13/go.mod h1:7S9/ev0klgBDR4GtXTXX8a3vIGJpMovkB8vQcUbaXHg=
github.com/mattn/go-ieproxy v0.0.1 h1:qiyop7gCflfhwCzGyeT0gro3sF9AIg9HU98JORTkqfI=
github.com/mattn/go-isatty v0.0.3/go.mod h1:M+lRXTBqGeGNdLjl/ufCoiOlB5xdOkqRJdNxMWT7Zi4=
github.com/mattn/go-isatty v0.0.4/go.mod h1:M+lRXTBqGeGNdLjl/ufCoiOlB5xdOkqRJdNxMWT7Zi4=
+github.com/mattn/go-isatty v0.0.8/go.mod h1:Iq45c/XA43vh69/j3iqttzPXn0bhXyGjM0Hdxcsrc5s=
+github.com/mattn/go-isatty v0.0.11/go.mod h1:PhnuNfih5lzO57/f3n+odYbM4JtupLOxQOAqxQCu2WE=
+github.com/mattn/go-isatty v0.0.12/go.mod h1:cbi8OIDigv2wuxKPP5vlRcQ1OAZbq2CE4Kysco4FUpU=
+github.com/mattn/go-isatty v0.0.14/go.mod h1:7GGIvUiUoEMVVmxf/4nioHXj79iQHKdU27kJ6hsGG94=
github.com/mattn/go-isatty v0.0.16/go.mod h1:kYGgaQfpe5nmfYZH+SKPsOc2e4SrIfOl2e/yFXSvRLM=
+github.com/mattn/go-isatty v0.0.17/go.mod h1:kYGgaQfpe5nmfYZH+SKPsOc2e4SrIfOl2e/yFXSvRLM=
github.com/mattn/go-isatty v0.0.19 h1:JITubQf0MOLdlGRuRq+jtsDlekdYPia9ZFsB8h/APPA=
github.com/mattn/go-isatty v0.0.19/go.mod h1:W+V8PltTTMOvKvAeJH7IuucS94S2C6jfK/D7dTCTo3Y=
github.com/mattn/go-oci8 v0.1.1/go.mod h1:wjDx6Xm9q7dFtHJvIlrI99JytznLw5wQ4R+9mNXJwGI=
@@ -1500,34 +1324,28 @@ github.com/mattn/go-runewidth v0.0.9/go.mod h1:H031xJmbD/WCDINGzjvQ9THkh0rPKHF+m
github.com/mattn/go-runewidth v0.0.13/go.mod h1:Jdepj2loyihRzMpdS35Xk/zdY8IAYHsh153qUoGf23w=
github.com/mattn/go-runewidth v0.0.14 h1:+xnbZSEeDbOIg5/mE6JF0w6n9duR1l3/WmbinWVwUuU=
github.com/mattn/go-runewidth v0.0.14/go.mod h1:Jdepj2loyihRzMpdS35Xk/zdY8IAYHsh153qUoGf23w=
-github.com/mattn/go-shellwords v1.0.3/go.mod h1:3xCvwCdWdlDJUrvuMn7Wuy9eWs4pE8vqg+NOMyg4B2o=
-github.com/mattn/go-shellwords v1.0.6/go.mod h1:3xCvwCdWdlDJUrvuMn7Wuy9eWs4pE8vqg+NOMyg4B2o=
github.com/mattn/go-shellwords v1.0.12 h1:M2zGm7EW6UQJvDeQxo4T51eKPurbeFbe8WtebGE2xrk=
github.com/mattn/go-shellwords v1.0.12/go.mod h1:EZzvwXDESEeg03EKmM+RmDnNOPKG4lLtQsUlTZDWQ8Y=
github.com/mattn/go-sqlite3 v1.6.0/go.mod h1:FPy6KqzDD04eiIsT53CuJW3U88zkxoIYsOqkbpncsNc=
-github.com/mattn/go-sqlite3 v1.11.0/go.mod h1:FPy6KqzDD04eiIsT53CuJW3U88zkxoIYsOqkbpncsNc=
github.com/mattn/go-sqlite3 v1.14.6/go.mod h1:NyWgC/yNuGj7Q9rpYnZvas74GogHl5/Z4A/KQRfk6bU=
-github.com/mattn/go-sqlite3 v1.14.14/go.mod h1:NyWgC/yNuGj7Q9rpYnZvas74GogHl5/Z4A/KQRfk6bU=
+github.com/mattn/go-sqlite3 v1.14.15/go.mod h1:2eHXhiwb8IkHr+BDWZGa96P6+rkvnG63S2DGjv9HUNg=
github.com/mattn/go-sqlite3 v2.0.3+incompatible h1:gXHsfypPkaMZrKbD5209QV9jbUTJKjyR5WD3HYQSd+U=
github.com/mattn/go-sqlite3 v2.0.3+incompatible/go.mod h1:FPy6KqzDD04eiIsT53CuJW3U88zkxoIYsOqkbpncsNc=
github.com/matttproud/golang_protobuf_extensions v1.0.1/go.mod h1:D8He9yQNgCq6Z5Ld7szi9bcBfOoFv/3dc6xSMkL2PC0=
-github.com/matttproud/golang_protobuf_extensions v1.0.2-0.20181231171920-c182affec369/go.mod h1:BSXmuO+STAnVfrANrmjBb36TMTDstsz7MSK+HVaYKv4=
github.com/matttproud/golang_protobuf_extensions v1.0.4 h1:mmDVorXM7PCGKw94cs5zkfA9PSy5pEvNWRP0ET0TIVo=
github.com/matttproud/golang_protobuf_extensions v1.0.4/go.mod h1:BSXmuO+STAnVfrANrmjBb36TMTDstsz7MSK+HVaYKv4=
-github.com/maxbrunsfeld/counterfeiter/v6 v6.2.2/go.mod h1:eD9eIE7cdwcMi9rYluz88Jz2VyhSmden33/aXg4oVIY=
github.com/miekg/dns v1.0.14/go.mod h1:W1PPwlIAgtquWBMBEV9nkV9Cazfe8ScdGz/Lj7v3Nrg=
+github.com/miekg/dns v1.1.50 h1:DQUfb9uc6smULcREF09Uc+/Gd46YWqJd5DbpPE9xkcA=
github.com/miekg/pkcs11 v1.0.2/go.mod h1:XsNlhZGX73bx86s2hdc/FuaLm2CPZJemRLMA+WTFxgs=
-github.com/miekg/pkcs11 v1.0.3/go.mod h1:XsNlhZGX73bx86s2hdc/FuaLm2CPZJemRLMA+WTFxgs=
github.com/miekg/pkcs11 v1.1.1 h1:Ugu9pdy6vAYku5DEpVWVFPYnzV+bxB+iRdbuFSu7TvU=
github.com/miekg/pkcs11 v1.1.1/go.mod h1:XsNlhZGX73bx86s2hdc/FuaLm2CPZJemRLMA+WTFxgs=
github.com/minio/md5-simd v1.1.2 h1:Gdi1DZK69+ZVMoNHRXJyNcxrMA4dSxoYHZSQbirFg34=
github.com/minio/minio-go/v7 v7.0.23 h1:NleyGQvAn9VQMU+YHVrgV4CX+EPtxPt/78lHOOTncy4=
github.com/minio/sha256-simd v1.0.0 h1:v1ta+49hkWZyvaKwrQB8elexRqm6Y0aMLjCNsrYxo6g=
-github.com/mistifyio/go-zfs v2.1.2-0.20190413222219-f784269be439+incompatible/go.mod h1:8AuVvqP/mXw1px98n46wfvcGfQ4ci2FwoAjKYxuo3Z4=
-github.com/mistifyio/go-zfs/v3 v3.0.0 h1:J5QK618xRcXnQYZ2GE5FdmpS1ufIrWue+lR/mpe6/14=
-github.com/mistifyio/go-zfs/v3 v3.0.0/go.mod h1:CzVgeB0RvF2EGzQnytKVvVSDwmKJXxkOTUGbNrTja/k=
+github.com/mistifyio/go-zfs/v3 v3.0.1 h1:YaoXgBePoMA12+S1u/ddkv+QqxcfiZK4prI6HPnkFiU=
+github.com/mistifyio/go-zfs/v3 v3.0.1/go.mod h1:CzVgeB0RvF2EGzQnytKVvVSDwmKJXxkOTUGbNrTja/k=
github.com/mitchellh/cli v1.0.0/go.mod h1:hNIlj7HEI86fIcpObd7a0FcrxTWetlwJDGcceTlRvqc=
-github.com/mitchellh/cli v1.1.4/go.mod h1:vTLESy5mRhKOs9KDp0/RATawxP1UqBmdrpVRMnpcvKQ=
+github.com/mitchellh/cli v1.1.5/go.mod h1:v8+iFts2sPIKUV1ltktPXMCC8fumSKFItNcD2cLtRR4=
github.com/mitchellh/copystructure v1.0.0/go.mod h1:SNtv71yrdKgLRyLFxmLdkAbkKEFWgYaq1OVrnRcwhnw=
github.com/mitchellh/copystructure v1.2.0 h1:vpKXTN4ewci03Vljg/q9QvCGUDttBOGBIa15WveJJGw=
github.com/mitchellh/copystructure v1.2.0/go.mod h1:qLl+cE2AmVv+CoeAwDPye/v+N2HKCj9FbZEVFJRxO9s=
@@ -1555,21 +1373,18 @@ github.com/mitchellh/reflectwalk v1.0.2/go.mod h1:mSTlrgnPZtwu0c4WaC2kGObEpuNDbx
github.com/mmcloughlin/avo v0.5.0/go.mod h1:ChHFdoV7ql95Wi7vuq2YT1bwCJqiWdZrQ1im3VujLYM=
github.com/moby/locker v1.0.1 h1:fOXqR41zeveg4fFODix+1Ch4mj/gT0NE1XJbp/epuBg=
github.com/moby/locker v1.0.1/go.mod h1:S7SDdo5zpBK84bzzVlKr2V0hz+7x9hWbYC/kq7oQppc=
-github.com/moby/moby v20.10.14+incompatible h1:J47P0p+O49F3au8QyE34dE/qXz571kcVmsbx8bvEuS0=
-github.com/moby/moby v20.10.14+incompatible/go.mod h1:fDXVQ6+S340veQPv35CzDahGBmHsiclFwfEygB/TWMc=
+github.com/moby/moby v24.0.6+incompatible h1:O/XZsZtaOVTYszsJQlr9pN1Zo1aRSH0KCWAIa6Kpm3s=
+github.com/moby/moby v24.0.6+incompatible/go.mod h1:fDXVQ6+S340veQPv35CzDahGBmHsiclFwfEygB/TWMc=
+github.com/moby/patternmatcher v0.6.0 h1:GmP9lR19aU5GqSSFko+5pRqHi+Ohk1O69aFiKkVGiPk=
+github.com/moby/patternmatcher v0.6.0/go.mod h1:hDPoyOpDY7OrrMDLaYoY3hf52gNCR/YOUYxkhApJIxc=
github.com/moby/spdystream v0.2.0 h1:cjW1zVyyoiM0T7b6UoySUFqzXMoqRckQtXwGPiBhOM8=
github.com/moby/spdystream v0.2.0/go.mod h1:f7i0iNDQJ059oMTcWxx8MA/zKFIuD/lY+0GqbN2Wy8c=
-github.com/moby/sys/mount v0.3.0 h1:bXZYMmq7DBQPwHRxH/MG+u9+XF90ZOwoXpHTOznMGp0=
-github.com/moby/sys/mount v0.3.0/go.mod h1:U2Z3ur2rXPFrFmy4q6WMwWrBOAQGYtYTRVM8BIvzbwk=
-github.com/moby/sys/mountinfo v0.4.0/go.mod h1:rEr8tzG/lsIZHBtN/JjGG+LMYx9eXgW2JI+6q0qou+A=
-github.com/moby/sys/mountinfo v0.4.1/go.mod h1:rEr8tzG/lsIZHBtN/JjGG+LMYx9eXgW2JI+6q0qou+A=
-github.com/moby/sys/mountinfo v0.5.0/go.mod h1:3bMD3Rg+zkqx8MRYPi7Pyb0Ie97QEBmdxbhnCLlSvSU=
github.com/moby/sys/mountinfo v0.6.2 h1:BzJjoreD5BMFNmD9Rus6gdd1pLuecOFPt8wC+Vygl78=
github.com/moby/sys/mountinfo v0.6.2/go.mod h1:IJb6JQeOklcdMU9F5xQ8ZALD+CUr5VlGpwtX+VE0rpI=
-github.com/moby/sys/symlink v0.1.0/go.mod h1:GGDODQmbFOjFsXvfLVn3+ZRxkch54RkSiGqsZeMYowQ=
-github.com/moby/term v0.0.0-20200312100748-672ec06f55cd/go.mod h1:DdlQx2hp0Ss5/fLikoLlEeIYiATotOjgB//nb973jeo=
-github.com/moby/term v0.0.0-20221205130635-1aeaba878587 h1:HfkjXDfhgVaN5rmueG8cL8KKeFNecRCXFhaJ2qZ5SKA=
-github.com/moby/term v0.0.0-20221205130635-1aeaba878587/go.mod h1:8FzsFHVUBGZdbDsJw/ot+X+d5HLUbvklYLJ9uGfcI3Y=
+github.com/moby/sys/sequential v0.5.0 h1:OPvI35Lzn9K04PBbCLW0g4LcFAJgHsvXsRyewg5lXtc=
+github.com/moby/sys/sequential v0.5.0/go.mod h1:tH2cOOs5V9MlPiXcQzRC+eEyab644PWKGRYaaV5ZZlo=
+github.com/moby/term v0.5.0 h1:xt8Q1nalod/v7BqbG21f8mQPqH+xAaC9C3N3wfWbVP0=
+github.com/moby/term v0.5.0/go.mod h1:8FzsFHVUBGZdbDsJw/ot+X+d5HLUbvklYLJ9uGfcI3Y=
github.com/modern-go/concurrent v0.0.0-20180228061459-e0a39a4cb421/go.mod h1:6dJC0mAP4ikYIbvyc7fijjWJddQyLn8Ig3JB5CqoB9Q=
github.com/modern-go/concurrent v0.0.0-20180306012644-bacd9c7ef1dd h1:TRLaZ9cD/w8PVh93nsPXa1VrQ6jlwL5oN8l14QlcNfg=
github.com/modern-go/concurrent v0.0.0-20180306012644-bacd9c7ef1dd/go.mod h1:6dJC0mAP4ikYIbvyc7fijjWJddQyLn8Ig3JB5CqoB9Q=
@@ -1588,81 +1403,56 @@ github.com/morikuni/aec v1.0.0 h1:nP9CBfwrvYnBRgY6qfDQkygYDmYwOilePFkwzv4dU8A=
github.com/morikuni/aec v1.0.0/go.mod h1:BbKIizmSmc5MMPqRYbxO4ZU0S0+P200+tUnFx7PXmsc=
github.com/mpvl/unique v0.0.0-20150818121801-cbe035fff7de h1:D5x39vF5KCwKQaw+OC9ZPiLVHXz3UFw2+psEX+gYcto=
github.com/mpvl/unique v0.0.0-20150818121801-cbe035fff7de/go.mod h1:kJun4WP5gFuHZgRjZUWWuH1DTxCtxbHDOIJsudS8jzY=
-github.com/mrunalp/fileutils v0.5.0/go.mod h1:M1WthSahJixYnrXQl/DFQuteStB1weuxD2QJNHXfbSQ=
github.com/mtibben/percent v0.2.1 h1:5gssi8Nqo8QU/r2pynCm+hBQHpkB/uNK7BJCFogWdzs=
github.com/mtibben/percent v0.2.1/go.mod h1:KG9uO+SZkUp+VkRHsCdYQV3XSZrrSpR3O9ibNBTZrns=
github.com/munnerz/goautoneg v0.0.0-20120707110453-a547fc61f48d/go.mod h1:+n7T8mK8HuQTcFwEeznm/DIxMOiR9yIdICNftLE1DvQ=
github.com/munnerz/goautoneg v0.0.0-20191010083416-a7dc8b61c822 h1:C3w9PqII01/Oq1c1nUAm88MOHcQC9l5mIlSMApZMrHA=
github.com/munnerz/goautoneg v0.0.0-20191010083416-a7dc8b61c822/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-20190716064945-2f068394615f/go.mod h1:qRWi+5nqEBWmkhHvq77mSJWrCKwh8bxhgT7d/eI7P4U=
github.com/mxk/go-flowrate v0.0.0-20140419014527-cca7078d478f h1:y5//uYreIhSUg3J1GEMiLbxo1LJaP8RfCpH6pymGZus=
github.com/mxk/go-flowrate v0.0.0-20140419014527-cca7078d478f/go.mod h1:ZdcZmHo+o7JKHSa8/e818NopupXU1YMK5fe1lsApnBw=
github.com/natefinch/atomic v1.0.1 h1:ZPYKxkqQOx3KZ+RsbnP/YsgvxWQPGxjC0oBt2AhwV0A=
+github.com/nelsam/hel/v2 v2.3.2/go.mod h1:1ZTGfU2PFTOd5mx22i5O0Lc2GY933lQ2wb/ggy+rL3w=
+github.com/nelsam/hel/v2 v2.3.3/go.mod h1:1ZTGfU2PFTOd5mx22i5O0Lc2GY933lQ2wb/ggy+rL3w=
github.com/niemeyer/pretty v0.0.0-20200227124842-a10e7caefd8e/go.mod h1:zD1mROLANZcx1PVRCS0qkT7pwLkGfwJo4zjcN/Tysno=
github.com/nsf/termbox-go v0.0.0-20190121233118-02980233997d h1:x3S6kxmy49zXVVyhcnrFqxvNVCBPb2KZ9hV2RBdS840=
github.com/nsf/termbox-go v0.0.0-20190121233118-02980233997d/go.mod h1:IuKpRQcYE1Tfu+oAQqaLisqDeXgjyyltCfsaoYN18NQ=
-github.com/nxadm/tail v1.4.4/go.mod h1:kenIhsEOeOJmVchQTgglprH7qJGnHDVpk1VPCcaMI8A=
github.com/nxadm/tail v1.4.8 h1:nPr65rt6Y5JFSKQO7qToXr7pePgD6Gwiw05lkbyAQTE=
-github.com/oklog/run v1.1.0 h1:GEenZ1cK0+q0+wsJew9qUg/DyD8k3JzYsZAi5gYi2mA=
+github.com/oklog/run v1.0.0 h1:Ru7dDtJNOyC66gQ5dQmaCa0qIsAUFY3sFpK1Xk8igrw=
github.com/oklog/ulid v1.3.1 h1:EGfNDEx6MqHz8B3uNV6QAib1UR2Lm97sHi3ocA6ESJ4=
github.com/oklog/ulid v1.3.1/go.mod h1:CirwcVhetQ6Lv90oh/F+FBtV6XMibvdAFo93nm5qn4U=
-github.com/olekukonko/tablewriter v0.0.0-20170122224234-a0225b3f23b5/go.mod h1:vsDQFd/mU46D+Z4whnwzcISnGGzXWMclvtLoiIKAKIo=
-github.com/olekukonko/tablewriter v0.0.5 h1:P2Ga83D34wi1o9J6Wh1mRuqd4mF/x/lgBS7N7AbDhec=
github.com/olekukonko/tablewriter v0.0.5/go.mod h1:hPp6KlRPjbx+hW8ykQs1w3UBbZlj6HuIJcUGPhkA7kY=
-github.com/onsi/ginkgo v0.0.0-20151202141238-7f8ab55aaf3b/go.mod h1:lLunBs/Ym6LB5Z9jYTR76FiuTmxDTDusOGeTQH+WWjE=
github.com/onsi/ginkgo v0.0.0-20170829012221-11459a886d9c/go.mod h1:lLunBs/Ym6LB5Z9jYTR76FiuTmxDTDusOGeTQH+WWjE=
github.com/onsi/ginkgo v1.6.0/go.mod h1:lLunBs/Ym6LB5Z9jYTR76FiuTmxDTDusOGeTQH+WWjE=
-github.com/onsi/ginkgo v1.8.0/go.mod h1:lLunBs/Ym6LB5Z9jYTR76FiuTmxDTDusOGeTQH+WWjE=
github.com/onsi/ginkgo v1.10.1/go.mod h1:lLunBs/Ym6LB5Z9jYTR76FiuTmxDTDusOGeTQH+WWjE=
-github.com/onsi/ginkgo v1.10.3/go.mod h1:lLunBs/Ym6LB5Z9jYTR76FiuTmxDTDusOGeTQH+WWjE=
github.com/onsi/ginkgo v1.11.0/go.mod h1:lLunBs/Ym6LB5Z9jYTR76FiuTmxDTDusOGeTQH+WWjE=
github.com/onsi/ginkgo v1.12.0/go.mod h1:oUhWkIvk5aDxtKvDDuw8gItl8pKl42LzjC9KZE0HfGg=
-github.com/onsi/ginkgo v1.12.1/go.mod h1:zj2OWP4+oCPe1qIXoGWkgMRwljMUYCdkwsT2108oapk=
github.com/onsi/ginkgo v1.16.5 h1:8xi0RTUf59SOSfEtZMvwTvXYMzG4gV23XVHOZiXNtnE=
github.com/onsi/ginkgo/v2 v2.11.0 h1:WgqUCUt/lT6yXoQ8Wef0fsNn5cAuMK7+KT9UFRz2tcU=
github.com/onsi/ginkgo/v2 v2.11.0/go.mod h1:ZhrRA5XmEE3x3rhlzamx/JJvujdZoJ2uvgI7kR0iZvM=
-github.com/onsi/gomega v0.0.0-20151007035656-2152b45fa28a/go.mod h1:C1qb7wdrVGGVU+Z6iS04AVkA3Q65CEZX59MT0QO5uiA=
github.com/onsi/gomega v0.0.0-20170829124025-dcabb60a477c/go.mod h1:C1qb7wdrVGGVU+Z6iS04AVkA3Q65CEZX59MT0QO5uiA=
-github.com/onsi/gomega v1.5.0/go.mod h1:ex+gbHU/CVuBBDIJjb2X0qEXbFg53c61hWP/1CpauHY=
github.com/onsi/gomega v1.7.0/go.mod h1:ex+gbHU/CVuBBDIJjb2X0qEXbFg53c61hWP/1CpauHY=
github.com/onsi/gomega v1.7.1/go.mod h1:XdKZgCCFLUoM/7CFJVPcG8C1xQ1AJ0vpAezJrB7JYyY=
github.com/onsi/gomega v1.9.0/go.mod h1:Ho0h+IUsWyvy1OpqCwxlQ/21gkhVunqlU8fDGcoTdcA=
-github.com/onsi/gomega v1.10.3/go.mod h1:V9xEwhxec5O8UDM77eCW8vLymOMltsqPVYWrpDsH8xc=
github.com/onsi/gomega v1.27.8 h1:gegWiwZjBsf2DgiSbf5hpokZ98JVDMcWkUiigk6/KXc=
github.com/onsi/gomega v1.27.8/go.mod h1:2J8vzI/s+2shY9XHRApDkdgPo1TKT7P2u6fXeJKFnNQ=
github.com/opencontainers/go-digest v0.0.0-20170106003457-a6d0ee40d420/go.mod h1:cMLVZDEM3+U2I4VmLI6N8jQYUd2OVphdqWwCJHrFt2s=
github.com/opencontainers/go-digest v0.0.0-20180430190053-c9281466c8b2/go.mod h1:cMLVZDEM3+U2I4VmLI6N8jQYUd2OVphdqWwCJHrFt2s=
-github.com/opencontainers/go-digest v1.0.0-rc1/go.mod h1:cMLVZDEM3+U2I4VmLI6N8jQYUd2OVphdqWwCJHrFt2s=
-github.com/opencontainers/go-digest v1.0.0-rc1.0.20180430190053-c9281466c8b2/go.mod h1:cMLVZDEM3+U2I4VmLI6N8jQYUd2OVphdqWwCJHrFt2s=
github.com/opencontainers/go-digest v1.0.0 h1:apOUWs51W5PlhuyGyz9FCeeBIOUDA/6nW8Oi/yOhh5U=
github.com/opencontainers/go-digest v1.0.0/go.mod h1:0JzlMkj0TRzQZfJkVvzbP0HBR3IKzErnv2BNG4W4MAM=
github.com/opencontainers/image-spec v1.0.1/go.mod h1:BtxoFyWECRxE4U/7sNtV5W15zMzWCbyJoFRP3s7yZA0=
github.com/opencontainers/image-spec v1.0.2/go.mod h1:BtxoFyWECRxE4U/7sNtV5W15zMzWCbyJoFRP3s7yZA0=
-github.com/opencontainers/image-spec v1.1.0-rc2 h1:2zx/Stx4Wc5pIPDvIxHXvXtQFW/7XWJGmnM7r3wg034=
-github.com/opencontainers/image-spec v1.1.0-rc2/go.mod h1:3OVijpioIKYWTqjiG0zfF6wvoJ4fAXGbjdZuI2NgsRQ=
+github.com/opencontainers/image-spec v1.1.0-rc5 h1:Ygwkfw9bpDvs+c9E34SdgGOj41dX/cbdlwvlWt0pnFI=
+github.com/opencontainers/image-spec v1.1.0-rc5/go.mod h1:X4pATf0uXsnn3g5aiGIsVnJBR4mxhKzfwmvK/B2NTm8=
github.com/opencontainers/runc v0.0.0-20190115041553-12f6a991201f/go.mod h1:qT5XzbpPznkRYVz/mWwUaVBUv2rmF59PVA73FjuZG0U=
github.com/opencontainers/runc v0.1.1/go.mod h1:qT5XzbpPznkRYVz/mWwUaVBUv2rmF59PVA73FjuZG0U=
-github.com/opencontainers/runc v1.0.0-rc8.0.20190926000215-3e425f80a8c9/go.mod h1:qT5XzbpPznkRYVz/mWwUaVBUv2rmF59PVA73FjuZG0U=
-github.com/opencontainers/runc v1.0.0-rc9/go.mod h1:qT5XzbpPznkRYVz/mWwUaVBUv2rmF59PVA73FjuZG0U=
-github.com/opencontainers/runc v1.0.0-rc93/go.mod h1:3NOsor4w32B2tC0Zbl8Knk4Wg84SM2ImC1fxBuqJ/H0=
-github.com/opencontainers/runc v1.0.2/go.mod h1:aTaHFFwQXuA71CiyxOdFFIorAoemI04suvGRQFzWTD0=
-github.com/opencontainers/runc v1.1.5 h1:L44KXEpKmfWDcS02aeGm8QNTFXTo2D+8MYGDIJ/GDEs=
-github.com/opencontainers/runc v1.1.5/go.mod h1:1J5XiS+vdZ3wCyZybsuxXZWGrgSr8fFJHLXuG2PsnNg=
-github.com/opencontainers/runtime-spec v0.1.2-0.20190507144316-5b71a03e2700/go.mod h1:jwyrGlmzljRJv/Fgzds9SsS/C5hL+LL3ko9hs6T5lQ0=
-github.com/opencontainers/runtime-spec v1.0.1/go.mod h1:jwyrGlmzljRJv/Fgzds9SsS/C5hL+LL3ko9hs6T5lQ0=
-github.com/opencontainers/runtime-spec v1.0.2-0.20190207185410-29686dbc5559/go.mod h1:jwyrGlmzljRJv/Fgzds9SsS/C5hL+LL3ko9hs6T5lQ0=
+github.com/opencontainers/runc v1.1.7 h1:y2EZDS8sNng4Ksf0GUYNhKbTShZJPJg1FiXJNH/uoCk=
+github.com/opencontainers/runc v1.1.7/go.mod h1:CbUumNnWCuTGFukNXahoo/RFBZvDAgRh/smNYNOhA50=
github.com/opencontainers/runtime-spec v1.0.2/go.mod h1:jwyrGlmzljRJv/Fgzds9SsS/C5hL+LL3ko9hs6T5lQ0=
-github.com/opencontainers/runtime-spec v1.0.3-0.20200929063507-e6143ca7d51d/go.mod h1:jwyrGlmzljRJv/Fgzds9SsS/C5hL+LL3ko9hs6T5lQ0=
-github.com/opencontainers/runtime-spec v1.0.3-0.20210326190908-1c3f411f0417 h1:3snG66yBm59tKhhSPQrQ/0bCrv1LQbKt40LnUPiUxdc=
-github.com/opencontainers/runtime-spec v1.0.3-0.20210326190908-1c3f411f0417/go.mod h1:jwyrGlmzljRJv/Fgzds9SsS/C5hL+LL3ko9hs6T5lQ0=
-github.com/opencontainers/runtime-tools v0.0.0-20181011054405-1d69bd0f9c39/go.mod h1:r3f7wjNzSs2extwzU3Y+6pKfobzPh+kKFJ3ofN+3nfs=
-github.com/opencontainers/selinux v1.6.0/go.mod h1:VVGKuOLlE7v4PJyT6h7mNWvq1rzqiriPsEqVhc+svHE=
-github.com/opencontainers/selinux v1.8.0/go.mod h1:RScLhm78qiWa2gbVCcGkC7tCGdgk3ogry1nUQF8Evvo=
-github.com/opencontainers/selinux v1.8.2/go.mod h1:MUIHuUEvKB1wtJjQdOyYRgOnLD2xAPP8dBsCoU0KuF8=
-github.com/opencontainers/selinux v1.10.0/go.mod h1:2i0OySw99QjzBBQByd1Gr9gSjvuho1lHsJxIJ3gGbJI=
-github.com/opencontainers/selinux v1.10.2 h1:NFy2xCsjn7+WspbfZkUd5zyVeisV7VFbPSP96+8/ha4=
-github.com/opencontainers/selinux v1.10.2/go.mod h1:cARutUbaUrlRClyvxOICCgKixCs6L05aUsohzA3EkHQ=
+github.com/opencontainers/runtime-spec v1.1.0-rc.3 h1:l04uafi6kxByhbxev7OWiuUv0LZxEsYUfDWZ6bztAuU=
+github.com/opencontainers/runtime-spec v1.1.0-rc.3/go.mod h1:jwyrGlmzljRJv/Fgzds9SsS/C5hL+LL3ko9hs6T5lQ0=
+github.com/opencontainers/selinux v1.11.0 h1:+5Zbo97w3Lbmb3PeqQtpmTkMwsW5nRI3YaLpt7tQ7oU=
+github.com/opencontainers/selinux v1.11.0/go.mod h1:E5dMC3VPuVvVHDYmi78qvhJp8+M586T4DlDRYpFkyec=
github.com/opentracing/opentracing-go v1.1.0/go.mod h1:UkNAQd3GIcIGf0SeVgPpRdFStlNbqXla1AfSYxPUl2o=
github.com/opentracing/opentracing-go v1.2.0 h1:uEJPy/1a5RIPAJ0Ov+OIO8OxWu77jEv+1B0VhjKrZUs=
github.com/ostreedev/ostree-go v0.0.0-20210805093236-719684c64e4f h1:/UDgs8FGMqwnHagNDPGOlts35QkhAZ8by3DR7nMih7M=
@@ -1672,7 +1462,6 @@ github.com/pashagolub/pgxmock/v2 v2.11.0 h1:ZUKqZy5Zf/5WJjAXHErjHngJBW5/3fEujGD+
github.com/pashagolub/pgxmock/v2 v2.11.0/go.mod h1:D3YslkN/nJ4+umVqWmbwfSXugJIjPMChkGBG47OJpNw=
github.com/pelletier/go-toml v1.2.0/go.mod h1:5z9KED0ma1S8pY6P1sdut58dfprrGBbd/94hg7ilaic=
github.com/pelletier/go-toml v1.7.0/go.mod h1:vwGMzjaWMwyfHwgIBhI2YUM4fB6nL6lVAvS1LBMMhTE=
-github.com/pelletier/go-toml v1.8.1/go.mod h1:T2/BmBdy8dvIRq1a/8aqjN41wvWlN4lrapLU/GW4pbc=
github.com/pelletier/go-toml v1.9.3/go.mod h1:u1nR/EPcESfeI/szUZKdtJ0xRNbUoANCkoOuaOx1Y+c=
github.com/pelletier/go-toml/v2 v2.0.8 h1:0ctb6s9mE31h0/lhu+J6OPmVeDxJn+kYnJc2jZR9tGQ=
github.com/pelletier/go-toml/v2 v2.0.8/go.mod h1:vuYfssBdrU2XDZ9bYydBu6t+6a6PYNcZljzZR9VXg+4=
@@ -1687,7 +1476,6 @@ github.com/pjbgf/sha1cd v0.3.0 h1:4D5XXmUUBUl/xQ6IjCkEAbqXskkq/4O7LmGn0AqMDs4=
github.com/pjbgf/sha1cd v0.3.0/go.mod h1:nZ1rrWOcGJ5uZgEEVL1VUM9iRQiZvWdbZjkKyFzPPsI=
github.com/pkg/diff v0.0.0-20210226163009-20ebb0f2a09e/go.mod h1:pJLUxLENpZxwdsKMEsNbx1VGcRFpLqf3715MtcvvzbA=
github.com/pkg/errors v0.8.0/go.mod h1:bwawxfHBFNV+L2hUp1rHADufV3IMtnDRdf1r5NINEl0=
-github.com/pkg/errors v0.8.1-0.20171018195549-f15c970de5b7/go.mod h1:bwawxfHBFNV+L2hUp1rHADufV3IMtnDRdf1r5NINEl0=
github.com/pkg/errors v0.8.1/go.mod h1:bwawxfHBFNV+L2hUp1rHADufV3IMtnDRdf1r5NINEl0=
github.com/pkg/errors v0.9.1 h1:FEBLx1zS214owpjy7qsBeixbURkuhQAwrK5UwLGTwt4=
github.com/pkg/errors v0.9.1/go.mod h1:bwawxfHBFNV+L2hUp1rHADufV3IMtnDRdf1r5NINEl0=
@@ -1697,11 +1485,12 @@ github.com/pkg/sftp v1.13.5/go.mod h1:wHDZ0IZX6JcBYRK1TH9bcVq8G7TLpVHYIGJRFnmPfx
github.com/pmezard/go-difflib v1.0.0 h1:4DBwDE0NGyQoBHbLQYPwSUPoCMWR5BEzIk/f1lZbAQM=
github.com/pmezard/go-difflib v1.0.0/go.mod h1:iKH77koFhYxTK1pcRnkKkqfTogsbg7gZNVY4sRDYZ/4=
github.com/posener/complete v1.1.1/go.mod h1:em0nMJCgc9GFtwrmVmEMR/ZL6WyhyjMBndrE9hABlRI=
+github.com/posener/complete v1.2.3/go.mod h1:WZIdtGGp+qx0sLrYKtIRAruyNpv6hFCicSgv7Sy7s/s=
github.com/power-devops/perfstat v0.0.0-20210106213030-5aafc221ea8c h1:ncq/mPwQF4JjgDlrVEn3C11VoGHZN7m8qihwgMEtzYw=
github.com/power-devops/perfstat v0.0.0-20210106213030-5aafc221ea8c/go.mod h1:OmDBASR4679mdNQnz2pUhc2G8CO2JrUAVFDRBDP/hJE=
-github.com/poy/onpar v0.0.0-20190519213022-ee068f8ea4d1 h1:oL4IBbcqwhhNWh31bjOX8C/OCy0zs9906d/VUru+bqg=
-github.com/poy/onpar v0.0.0-20190519213022-ee068f8ea4d1/go.mod h1:nSbFQvMj97ZyhFRSJYtut+msi4sOY6zJDGCdSc+/rZU=
-github.com/pquerna/cachecontrol v0.0.0-20171018203845-0dec1b30a021/go.mod h1:prYjPmNq4d1NPVmpShWobRqXY3q7Vp+80DqgxxUrUIA=
+github.com/poy/onpar v0.0.0-20200406201722-06f95a1c68e8/go.mod h1:nSbFQvMj97ZyhFRSJYtut+msi4sOY6zJDGCdSc+/rZU=
+github.com/poy/onpar v1.1.2 h1:QaNrNiZx0+Nar5dLgTVp5mXkyoVFIbepjyEoGSnhbAY=
+github.com/poy/onpar v1.1.2/go.mod h1:6X8FLNoxyr9kkmnlqpK6LSoiOtrO6MICtWwEuWkLjzg=
github.com/prashantv/gostub v1.1.0 h1:BTyx3RfQjRHnUWaGF9oQos79AlQ5k8WNktv7VGvVH4g=
github.com/proglottis/gpgme v0.1.3 h1:Crxx0oz4LKB3QXc5Ea0J19K/3ICfy3ftr5exgUK1AU0=
github.com/proglottis/gpgme v0.1.3/go.mod h1:fPbW/EZ0LvwQtH8Hy7eixhp1eF3G39dtx7GUN+0Gmy0=
@@ -1709,13 +1498,12 @@ github.com/prometheus-community/pro-bing v0.3.0 h1:SFT6gHqXwbItEDJhTkzPWVqU6CLEt
github.com/prometheus-community/pro-bing v0.3.0/go.mod h1:p9dLb9zdmv+eLxWfCT6jESWuDrS+YzpPkQBgysQF8a0=
github.com/prometheus/client_golang v0.9.0-pre1.0.20180209125602-c332b6f63c06/go.mod h1:7SWBe2y4D6OKWSNQJUaRYU/AaXPKyh/dDVn+NZz0KFw=
github.com/prometheus/client_golang v0.9.1/go.mod h1:7SWBe2y4D6OKWSNQJUaRYU/AaXPKyh/dDVn+NZz0KFw=
+github.com/prometheus/client_golang v0.9.2/go.mod h1:OsXs2jCmiKlQ1lTBmv21f2mNfw4xf/QclQDMrYNZzcM=
github.com/prometheus/client_golang v0.9.3/go.mod h1:/TN21ttK/J9q6uSwhBd54HahCDft0ttaMvbicHlPoso=
github.com/prometheus/client_golang v1.0.0/go.mod h1:db9x61etRT2tGnBNRi70OPL5FsnadC4Ky3P0J6CfImo=
github.com/prometheus/client_golang v1.1.0/go.mod h1:I1FGZT9+L76gKKOs5djB6ezCbFQP1xR9D75/vuwEF3g=
-github.com/prometheus/client_golang v1.7.1/go.mod h1:PY5Wy2awLA44sXw4AOSfFBetzPP4j5+D6mVACh+pe2M=
-github.com/prometheus/client_golang v1.11.1/go.mod h1:Z6t4BnS23TR94PD6BsDNk8yVqroYurpAkEiz0P2BEV0=
-github.com/prometheus/client_golang v1.15.1 h1:8tXpTmJbyH5lydzFPoxSIJ0J46jdh3tylbvM1xCv0LI=
-github.com/prometheus/client_golang v1.15.1/go.mod h1:e9yaBhRPU2pPNsZwE+JdQl0KEt1N9XgF6zxWmaC0xOk=
+github.com/prometheus/client_golang v1.16.0 h1:yk/hx9hDbrGHovbci4BY+pRMfSuuat626eFsHb7tmT8=
+github.com/prometheus/client_golang v1.16.0/go.mod h1:Zsulrv/L9oM40tJ7T815tM89lFEugiJ9HzIqaAx4LKc=
github.com/prometheus/client_model v0.0.0-20171117100541-99fa1f4be8e5/go.mod h1:MbSGuTsp3dbXC40dX6PRTWyKYBIrTGTE9sqQNg2J8bo=
github.com/prometheus/client_model v0.0.0-20180712105110-5c3871d89910/go.mod h1:MbSGuTsp3dbXC40dX6PRTWyKYBIrTGTE9sqQNg2J8bo=
github.com/prometheus/client_model v0.0.0-20190129233127-fd36f4220a90/go.mod h1:xMI15A0UPsDsEKsMN9yxemIoYk6Tm2C1GtYGdfGttqA=
@@ -1725,29 +1513,26 @@ github.com/prometheus/client_model v0.4.0 h1:5lQXD3cAg1OXBf4Wq03gTrXHeaV0TQvGfUo
github.com/prometheus/client_model v0.4.0/go.mod h1:oMQmHW1/JoDwqLtg57MGgP/Fb1CJEYF2imWWhWtMkYU=
github.com/prometheus/common v0.0.0-20180110214958-89604d197083/go.mod h1:daVV7qP5qjZbuso7PdcryaAu0sAZbrN9i7WWcTMWvro=
github.com/prometheus/common v0.0.0-20181113130724-41aa239b4cce/go.mod h1:daVV7qP5qjZbuso7PdcryaAu0sAZbrN9i7WWcTMWvro=
+github.com/prometheus/common v0.0.0-20181126121408-4724e9255275/go.mod h1:daVV7qP5qjZbuso7PdcryaAu0sAZbrN9i7WWcTMWvro=
github.com/prometheus/common v0.4.0/go.mod h1:TNfzLD0ON7rHzMJeJkieUDPYmFC7Snx/y86RQel1bk4=
github.com/prometheus/common v0.4.1/go.mod h1:TNfzLD0ON7rHzMJeJkieUDPYmFC7Snx/y86RQel1bk4=
github.com/prometheus/common v0.6.0/go.mod h1:eBmuwkDJBwy6iBfxCBob6t6dR6ENT/y+J+Zk0j9GMYc=
-github.com/prometheus/common v0.10.0/go.mod h1:Tlit/dnDKsSWFlCLTWaA1cyBgKHSMdTB80sz/V91rCo=
-github.com/prometheus/common v0.26.0/go.mod h1:M7rCNAaPfAosfx8veZJCuw84e35h3Cfd9VFqTh1DIvc=
-github.com/prometheus/common v0.42.0 h1:EKsfXEYo4JpWMHH5cg+KOUWeuJSov1Id8zGR8eeI1YM=
-github.com/prometheus/common v0.42.0/go.mod h1:xBwqVerjNdUDjgODMpudtOMwlOwf2SaTr1yjz4b7Zbc=
+github.com/prometheus/common v0.44.0 h1:+5BrQJwiBB9xsMygAB3TNvpQKOwlkc25LbISbrdOOfY=
+github.com/prometheus/common v0.44.0/go.mod h1:ofAIvZbQ1e/nugmZGz4/qCb9Ap1VoSTIO7x0VV9VvuY=
github.com/prometheus/procfs v0.0.0-20180125133057-cb4147076ac7/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-20181204211112-1dc9a6cbc91a/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-20190522114515-bc1a522cf7b1/go.mod h1:TjEm7ze935MbeOT/UhFTIMYKhuLP4wbCsTZCD3I8kEA=
github.com/prometheus/procfs v0.0.2/go.mod h1:TjEm7ze935MbeOT/UhFTIMYKhuLP4wbCsTZCD3I8kEA=
github.com/prometheus/procfs v0.0.3/go.mod h1:4A/X28fw3Fc593LaREMrKMqOKvUAntwMDaekg4FpcdQ=
-github.com/prometheus/procfs v0.0.5/go.mod h1:4A/X28fw3Fc593LaREMrKMqOKvUAntwMDaekg4FpcdQ=
-github.com/prometheus/procfs v0.0.8/go.mod h1:7Qr8sr6344vo1JqZ6HhLceV9o3AJ1Ff+GxbHq6oeK9A=
-github.com/prometheus/procfs v0.1.3/go.mod h1:lV6e/gmhEcM9IjHGsFOCxxuZ+z1YqCvr4OA4YeYWdaU=
-github.com/prometheus/procfs v0.2.0/go.mod h1:lV6e/gmhEcM9IjHGsFOCxxuZ+z1YqCvr4OA4YeYWdaU=
-github.com/prometheus/procfs v0.6.0/go.mod h1:cz+aTbrPOrUb4q7XlbU9ygM+/jj0fzG6c1xBZuNvfVA=
-github.com/prometheus/procfs v0.9.0 h1:wzCHvIvM5SxWqYvwgVL7yJY8Lz3PKn49KQtpgMYJfhI=
-github.com/prometheus/procfs v0.9.0/go.mod h1:+pB4zwohETzFnmlpe6yd2lSc+0/46IYZRB/chUwxUZY=
+github.com/prometheus/procfs v0.10.1 h1:kYK1Va/YMlutzCGazswoHKo//tZVlFpKYh+PymziUAg=
+github.com/prometheus/procfs v0.10.1/go.mod h1:nwNm2aOCAYw8uTR/9bWRREkZFxAUcWzPHWJq+XBB/FM=
github.com/prometheus/tsdb v0.7.1/go.mod h1:qhTCs0VvXwvX/y3TZrWD7rabWM+ijKTux40TwIPHuXU=
github.com/protocolbuffers/txtpbfmt v0.0.0-20230328191034-3462fbc510c0 h1:sadMIsgmHpEOGbUs6VtHBXRR1OHevnj7hLx9ZcdNGW4=
github.com/protocolbuffers/txtpbfmt v0.0.0-20230328191034-3462fbc510c0/go.mod h1:jgxiZysxFPM+iWKwQwPR+y+Jvo54ARd4EisXxKYpB5c=
+github.com/rancher/dynamiclistener v0.3.5 h1:5TaIHvkDGmZKvc96Huur16zfTKOiLhDtK4S+WV0JA6A=
+github.com/rancher/wharfie v0.6.2 h1:ZTrZ0suU0abWwLLf2zaqjhwpxK8+BkbnMocnU2u1bSQ=
+github.com/rancher/wharfie v0.6.2/go.mod h1:7ii0+eehBwUEFaJMiRHWCbvN11bsfVHT1oc+P/6IBSg=
github.com/rcrowley/go-metrics v0.0.0-20201227073835-cf1acfcdf475 h1:N/ElC8H3+5XpJzTSTfLsJV/mx9Q9g7kxmchpfZyxgzM=
github.com/rcrowley/go-metrics v0.0.0-20201227073835-cf1acfcdf475/go.mod h1:bCqnVzQkZxMG4s8nGwiZ5l3QUCyqpo9Y+/ZMZ9VjZe4=
github.com/redis/go-redis/v9 v9.0.5 h1:CuQcn5HIEeK7BgElubPP8CGtE0KakrnbBSTLjathl5o=
@@ -1759,8 +1544,8 @@ github.com/replicatedhq/troubleshoot v0.57.0/go.mod h1:R5VdixzaBXfWLbP9mcLuZKs/b
github.com/rifflock/lfshook v0.0.0-20180920164130-b9218ef580f5 h1:mZHayPoR0lNmnHyvtYjDeq0zlVHn9K/ZXoy17ylucdo=
github.com/rifflock/lfshook v0.0.0-20180920164130-b9218ef580f5/go.mod h1:GEXHk5HgEKCvEIIrSpFI3ozzG5xOKA2DVlEX/gGnewM=
github.com/rivo/uniseg v0.2.0/go.mod h1:J6wj4VEh+S6ZtnVlnTBMWIodfgj8LQOQFoIToxlJtxc=
-github.com/rivo/uniseg v0.4.3 h1:utMvzDsuh3suAEnhH0RdHmoPbU648o6CvXxTx4SBMOw=
-github.com/rivo/uniseg v0.4.3/go.mod h1:FN3SvrM+Zdj16jyLfmOkMNblXMcoc8DfTHruCPUcx88=
+github.com/rivo/uniseg v0.4.4 h1:8TfxU8dW6PdqD27gjM8MVNuicgxIjxpm4K7x4jp8sis=
+github.com/rivo/uniseg v0.4.4/go.mod h1:FN3SvrM+Zdj16jyLfmOkMNblXMcoc8DfTHruCPUcx88=
github.com/robfig/cron/v3 v3.0.1 h1:WdRxkvbJztn8LMz/QEvLN5sBU+xKpSqwwUO1Pjr4qDs=
github.com/robfig/cron/v3 v3.0.1/go.mod h1:eQICP3HwyT7UooqI/z+Ov+PtYAWygg1TEWWzGIFLtro=
github.com/rogpeppe/fastuuid v0.0.0-20150106093220-6724a57986af/go.mod h1:XWv6SoW27p1b0cqNHllgS5HIMJraePCO15w5zCzIWYg=
@@ -1768,26 +1553,23 @@ github.com/rogpeppe/fastuuid v1.2.0/go.mod h1:jVj6XXZzXRy/MSR5jhDC/2q6DgLz+nrA6L
github.com/rogpeppe/go-internal v1.1.0/go.mod h1:M8bDsm7K2OlrFYOpmOWEs/qY81heoFRclV5y23lUDJ4=
github.com/rogpeppe/go-internal v1.2.2/go.mod h1:M8bDsm7K2OlrFYOpmOWEs/qY81heoFRclV5y23lUDJ4=
github.com/rogpeppe/go-internal v1.3.0/go.mod h1:M8bDsm7K2OlrFYOpmOWEs/qY81heoFRclV5y23lUDJ4=
+github.com/rogpeppe/go-internal v1.6.1/go.mod h1:xXDCJY+GAPziupqXw64V24skbSoqbTEfhy4qGm1nDQc=
github.com/rogpeppe/go-internal v1.8.0/go.mod h1:WmiCO8CzOY8rg0OYDC4/i/2WRWAB6poM+XZ2dLUbcbE=
+github.com/rogpeppe/go-internal v1.9.0/go.mod h1:WtVeX8xhTBvf0smdhujwtBcq4Qrzq/fJaraNFVN+nFs=
github.com/rogpeppe/go-internal v1.10.0 h1:TMyTOH3F/DB16zRVcYyreMH6GnZZrwQVAoYjRBZyWFQ=
github.com/rs/xid v1.3.0 h1:6NjYksEUlhurdVehpc7S7dk6DAmcKv8V9gG0FsVN2U4=
-github.com/rubenv/sql-migrate v1.2.0 h1:fOXMPLMd41sK7Tg75SXDec15k3zg5WNV6SjuDRiNfcU=
-github.com/rubenv/sql-migrate v1.2.0/go.mod h1:Z5uVnq7vrIrPmHbVFfR4YLHRZquxeHpckCnRq0P/K9Y=
+github.com/rubenv/sql-migrate v1.3.1 h1:Vx+n4Du8X8VTYuXbhNxdEUoh6wiJERA0GlWocR5FrbA=
+github.com/rubenv/sql-migrate v1.3.1/go.mod h1:YzG/Vh82CwyhTFXy+Mf5ahAiiEOpAlHurg+23VEzcsk=
github.com/russross/blackfriday/v2 v2.0.1/go.mod h1:+Rmxgy9KzJVeS9/2gXHxylqXiyQDYRxCVz55jmeOWTM=
github.com/russross/blackfriday/v2 v2.1.0 h1:JIOH55/0cWyOuilr9/qlrm0BSXldqnqwMsf35Ld67mk=
github.com/russross/blackfriday/v2 v2.1.0/go.mod h1:+Rmxgy9KzJVeS9/2gXHxylqXiyQDYRxCVz55jmeOWTM=
github.com/ryanuber/columnize v0.0.0-20160712163229-9b3edd62028f/go.mod h1:sm1tb6uqfes/u+d4ooFouqFdy9/2g9QGwK3SQygK0Ts=
-github.com/safchain/ethtool v0.0.0-20190326074333-42ed695e3de8/go.mod h1:Z0q5wiBQGYcxhMZ6gUqHn6pYNLypFAvaL3UvgZLR0U4=
github.com/sahilm/fuzzy v0.1.0 h1:FzWGaw2Opqyu+794ZQ9SYifWv2EIXpwP4q8dY1kDAwI=
github.com/sahilm/fuzzy v0.1.0/go.mod h1:VFvziUEIMCrT6A6tw2RFIXPXXmzXbOsSHF0DOI8ZK9Y=
-github.com/satori/go.uuid v1.2.0/go.mod h1:dA0hQrYB0VpLJoorglMZABFdXlWrHn1NEOzdhQKdks0=
-github.com/sclevine/spec v1.2.0/go.mod h1:W4J29eT/Kzv7/b9IWLB055Z+qvVC9vt0Arko24q7p+U=
github.com/sean-/seed v0.0.0-20170313163322-e2103e2c3529/go.mod h1:DxrIzT+xaE7yg65j358z/aeFdxmN0P9QXhEzd20vsDc=
github.com/sebdah/goldie v1.0.0 h1:9GNhIat69MSlz/ndaBg48vl9dF5fI+NBB6kfOxgfkMc=
github.com/sebdah/goldie v1.0.0/go.mod h1:jXP4hmWywNEwZzhMuv2ccnqTSFpuq8iyQhtQdkkZBH4=
github.com/sebdah/goldie/v2 v2.5.3 h1:9ES/mNN+HNUbNWpVAlrzuZ7jE+Nrczbj8uFRjM7624Y=
-github.com/seccomp/libseccomp-golang v0.9.1/go.mod h1:GbW5+tmTXfcxTToHLXlScSlAvWlF4P2Ca7zGrPiEpWo=
-github.com/seccomp/libseccomp-golang v0.9.2-0.20220502022130-f33da4d89646/go.mod h1:JA8cRccbGaA1s33RQf7Y1+q9gHmZX1yB/z9WDN1C6fg=
github.com/segmentio/ksuid v1.0.4 h1:sBo2BdShXjmcugAMwjugoGUdUV0pcxY5mW4xKRn3v4c=
github.com/segmentio/ksuid v1.0.4/go.mod h1:/XUiZBD3kVx5SmUOl55voK5yeAbBNNIed+2O73XgrPE=
github.com/sergi/go-diff v1.1.0/go.mod h1:STckp+ISIX8hZLjrqAeVduY0gWCT9IjLuqbuNXdaHfM=
@@ -1801,16 +1583,16 @@ github.com/shoenig/go-m1cpu v0.1.6 h1:nxdKQNcEB6vzgA2E2bvzKIYRuNj7XNJ4S/aRSwKzFt
github.com/shoenig/go-m1cpu v0.1.6/go.mod h1:1JJMcUBvfNwpq05QDQVAnx3gUHr9IYF7GNg9SUEw2VQ=
github.com/shoenig/test v0.6.4 h1:kVTaSd7WLz5WZ2IaoM0RSzRsUD+m8wRR+5qvntpn4LU=
github.com/shoenig/test v0.6.4/go.mod h1:byHiCGXqrVaflBLAMq/srcZIHynQPQgeyvkvXnjqq0k=
-github.com/shopspring/decimal v1.2.0 h1:abSATXmQEYyShuxI4/vyW3tV1MrKAJzCZ/0zLUXYbsQ=
github.com/shopspring/decimal v1.2.0/go.mod h1:DKyhrW/HYNuLGql+MJL6WCR6knT2jwCFRcu2hWCYk4o=
+github.com/shopspring/decimal v1.3.1 h1:2Usl1nmF/WZucqkFZhnfFYxxxu8LG21F6nPQBE5gKV8=
+github.com/shopspring/decimal v1.3.1/go.mod h1:DKyhrW/HYNuLGql+MJL6WCR6knT2jwCFRcu2hWCYk4o=
github.com/shurcooL/sanitized_anchor_name v1.0.0/go.mod h1:1NzhyTcUVG4SuEtjjoZeVRXNmyL/1OwPU0+IJeTBvfc=
-github.com/sigstore/fulcio v1.0.0 h1:hBZW6qg9GXTtCX8jOg1hmyjYLrmsEKZGeMwAbW3XNEg=
-github.com/sigstore/fulcio v1.0.0/go.mod h1:j4MzLxX/Be0rHYh3JF2dgMorkWGzEMHBqIHwFU8I/Rw=
-github.com/sigstore/rekor v1.2.0 h1:ahlnoEY3zo8Vc+eZLPobamw6YfBTAbI0lthzUQd6qe4=
-github.com/sigstore/rekor v1.2.0/go.mod h1:zcFO54qIg2G1/i0sE/nvmELUOng/n0MPjTszRYByVPo=
-github.com/sigstore/sigstore v1.6.4 h1:jH4AzR7qlEH/EWzm+opSpxCfuUcjHL+LJPuQE7h40WE=
-github.com/sigstore/sigstore v1.6.4/go.mod h1:pjR64lBxnjoSrAr+Ydye/FV73IfrgtoYlAI11a8xMfA=
-github.com/sirupsen/logrus v1.0.4-0.20170822132746-89742aefa4b2/go.mod h1:pMByvHTf9Beacp5x1UXfOR9xyW/9antXMhjMPG0dEzc=
+github.com/sigstore/fulcio v1.3.1 h1:0ntW9VbQbt2JytoSs8BOGB84A65eeyvGSavWteYp29Y=
+github.com/sigstore/fulcio v1.3.1/go.mod h1:/XfqazOec45ulJZpyL9sq+OsVQ8g2UOVoNVi7abFgqU=
+github.com/sigstore/rekor v1.2.2-0.20230601122533-4c81ff246d12 h1:x/WnxasgR40qGY67IHwioakXLuhDxJ10vF8/INuOTiI=
+github.com/sigstore/rekor v1.2.2-0.20230601122533-4c81ff246d12/go.mod h1:8c+a8Yo7r8gKuYbIaz+c3oOdw9iMXx+tMdOg2+b+2jQ=
+github.com/sigstore/sigstore v1.7.1 h1:fCATemikcBK0cG4+NcM940MfoIgmioY1vC6E66hXxks=
+github.com/sigstore/sigstore v1.7.1/go.mod h1:0PmMzfJP2Y9+lugD0wer4e7TihR5tM7NcIs3bQNk5xg=
github.com/sirupsen/logrus v1.0.6/go.mod h1:pMByvHTf9Beacp5x1UXfOR9xyW/9antXMhjMPG0dEzc=
github.com/sirupsen/logrus v1.2.0/go.mod h1:LxeOpSwHxABJmUn/MG1IvRgCAasNZTLOkJPxbbu5VWo=
github.com/sirupsen/logrus v1.3.0/go.mod h1:LxeOpSwHxABJmUn/MG1IvRgCAasNZTLOkJPxbbu5VWo=
@@ -1820,6 +1602,7 @@ github.com/sirupsen/logrus v1.4.2/go.mod h1:tLMulIdttU9McNUspp0xgXVQah82FyeX6Mwd
github.com/sirupsen/logrus v1.6.0/go.mod h1:7uNnSEd1DgxDLC74fIahvMZmmYsHGZGEOFrfsX/uA88=
github.com/sirupsen/logrus v1.7.0/go.mod h1:yWOB1SBYBC5VeMP7gHvWumXLIWorT60ONWic61uBYv0=
github.com/sirupsen/logrus v1.8.1/go.mod h1:yWOB1SBYBC5VeMP7gHvWumXLIWorT60ONWic61uBYv0=
+github.com/sirupsen/logrus v1.9.0/go.mod h1:naHLuLoDiP4jHNo9R0sCBMtWGeIprob74mVsIT4qYEQ=
github.com/sirupsen/logrus v1.9.3 h1:dueUQJ1C2q9oE3F7wvmSGAaVtTmUizReu6fjN8uqzbQ=
github.com/sirupsen/logrus v1.9.3/go.mod h1:naHLuLoDiP4jHNo9R0sCBMtWGeIprob74mVsIT4qYEQ=
github.com/skeema/knownhosts v1.1.0 h1:Wvr9V0MxhjRbl3f9nMnKnFfiWTJmtECJ9Njkea3ysW0=
@@ -1838,14 +1621,14 @@ github.com/spf13/afero v1.2.2/go.mod h1:9ZxEEn6pIJ8Rxe320qSDBk6AsU0r9pR7Q4OcevTd
github.com/spf13/cast v0.0.0-20150508191742-4d07383ffe94/go.mod h1:r2rcYCSwa1IExKTDiTfzaxqT2FNHs8hODu4LnUfgKEg=
github.com/spf13/cast v1.3.0/go.mod h1:Qx5cxh0v+4UWYiBimWS+eyWzqEqokIECu5etghLkUJE=
github.com/spf13/cast v1.3.1/go.mod h1:Qx5cxh0v+4UWYiBimWS+eyWzqEqokIECu5etghLkUJE=
+github.com/spf13/cast v1.5.0/go.mod h1:SpXXQ5YoyJw6s3/6cMTQuxvgRl3PCJiyaX9p6b155UU=
github.com/spf13/cast v1.5.1 h1:R+kOtfhWQE6TVQzY+4D7wJLBgkdVasCEFxSUBYBYIlA=
github.com/spf13/cast v1.5.1/go.mod h1:b9PdjNptOpzXr7Rq1q9gJML/2cdGQAo69NKzQ10KN48=
github.com/spf13/cobra v0.0.1/go.mod h1:1l0Ry5zgKvJasoi3XT1TypsSe7PqH0Sj9dhYf7v3XqQ=
github.com/spf13/cobra v0.0.2-0.20171109065643-2da4a54c5cee/go.mod h1:1l0Ry5zgKvJasoi3XT1TypsSe7PqH0Sj9dhYf7v3XqQ=
github.com/spf13/cobra v0.0.3/go.mod h1:1l0Ry5zgKvJasoi3XT1TypsSe7PqH0Sj9dhYf7v3XqQ=
-github.com/spf13/cobra v1.0.0/go.mod h1:/6GTrnGXV9HjY+aR4k0oJ5tcvakLuG6EuKReYlHNrgE=
+github.com/spf13/cobra v0.0.6/go.mod h1:/6GTrnGXV9HjY+aR4k0oJ5tcvakLuG6EuKReYlHNrgE=
github.com/spf13/cobra v1.1.1/go.mod h1:WnodtKOvamDL/PwE2M4iKs8aMDBZ5Q5klgD3qfVJQMI=
-github.com/spf13/cobra v1.1.3/go.mod h1:pGADOWyqRD/YMrPZigI/zbliZ2wVD/23d+is3pSWzOo=
github.com/spf13/cobra v1.2.1/go.mod h1:ExllRjgxM/piMAM+3tAZvg8fsklGAf3tPfi+i8t68Nk=
github.com/spf13/cobra v1.7.0 h1:hyqWnYt1ZQShIddO5kBpj3vu05/++x6tJ6dg8EC572I=
github.com/spf13/cobra v1.7.0/go.mod h1:uLxZILRyS/50WlhOIKD7W6V5bgeIt+4sICxh6uRMrb0=
@@ -1856,7 +1639,6 @@ github.com/spf13/jwalterweatherman v1.1.0/go.mod h1:aNWZUN0dPAAO/Ljvb5BEdw96iTZ0
github.com/spf13/pflag v0.0.0-20170130214245-9ff6c6923cff/go.mod h1:DYY7MBk1bdzusC3SYhjObp+wFpr4gzcvqqNjLnInEg4=
github.com/spf13/pflag v1.0.0/go.mod h1:DYY7MBk1bdzusC3SYhjObp+wFpr4gzcvqqNjLnInEg4=
github.com/spf13/pflag v1.0.1-0.20171106142849-4c012f6dcd95/go.mod h1:DYY7MBk1bdzusC3SYhjObp+wFpr4gzcvqqNjLnInEg4=
-github.com/spf13/pflag v1.0.1/go.mod h1:DYY7MBk1bdzusC3SYhjObp+wFpr4gzcvqqNjLnInEg4=
github.com/spf13/pflag v1.0.3/go.mod h1:DYY7MBk1bdzusC3SYhjObp+wFpr4gzcvqqNjLnInEg4=
github.com/spf13/pflag v1.0.5 h1:iy+VFUOCP1a+8yFto/drg2CJ5u0yRoB7fZw3DKv/JXA=
github.com/spf13/pflag v1.0.5/go.mod h1:McXfInJRrz4CZXVZOBLb0bTZqETkiAhM9Iw0y3An2Bg=
@@ -1870,14 +1652,12 @@ github.com/stefanberger/go-pkcs11uri v0.0.0-20201008174630-78d3cae3a980 h1:lIOOH
github.com/stefanberger/go-pkcs11uri v0.0.0-20201008174630-78d3cae3a980/go.mod h1:AO3tvPzVZ/ayst6UlUKUv6rcPQInYe3IknH3jYhAKu8=
github.com/stoewer/go-strcase v1.2.0 h1:Z2iHWqGXH00XYgqDmNgQbIBxf3wrNq0F3feEy0ainaU=
github.com/stoewer/go-strcase v1.2.0/go.mod h1:IBiWB2sKIp3wVVQ3Y035++gc+knqhUQag1KpM8ahLw8=
-github.com/stretchr/objx v0.0.0-20180129172003-8a3f7159479f/go.mod h1:HFkY916IF+rwdDfMAkV7OtwuqBVzrE8GR6GFx+wExME=
github.com/stretchr/objx v0.1.0/go.mod h1:HFkY916IF+rwdDfMAkV7OtwuqBVzrE8GR6GFx+wExME=
github.com/stretchr/objx v0.1.1/go.mod h1:HFkY916IF+rwdDfMAkV7OtwuqBVzrE8GR6GFx+wExME=
github.com/stretchr/objx v0.2.0/go.mod h1:qt09Ya8vawLte6SNmTgCsAVtYtaKzEcn8ATUoHMkEqE=
github.com/stretchr/objx v0.4.0/go.mod h1:YvHI0jy2hoMjB+UWwv71VJQ9isScKT/TqJzVSSt89Yw=
github.com/stretchr/objx v0.5.0 h1:1zr/of2m5FGMsad5YfcqgdqdWrIhu+EBEJRhR1U7z/c=
github.com/stretchr/objx v0.5.0/go.mod h1:Yh+to48EsGEfYuaHDzXPcE3xhTkx73EhmCGUpEOglKo=
-github.com/stretchr/testify v0.0.0-20180303142811-b89eecf5ca5d/go.mod h1:a8OnRcib4nhh0OaRAV+Yts87kKdq0PP7pXfy6kDkUVs=
github.com/stretchr/testify v1.2.2/go.mod h1:a8OnRcib4nhh0OaRAV+Yts87kKdq0PP7pXfy6kDkUVs=
github.com/stretchr/testify v1.3.0/go.mod h1:M5WIy9Dh21IEIfnGCwXGc5bZfKNJtfHm1UVUgZn+9EI=
github.com/stretchr/testify v1.4.0/go.mod h1:j7eGeouHqKxXV5pUuKE4zz7dFj8WfuZ+81PSLYec5m4=
@@ -1896,15 +1676,12 @@ github.com/subosito/gotenv v1.4.2 h1:X1TuBLAMDFbaTAChgCBLu3DU3UPyELpnF2jjJ2cz/S8
github.com/subosito/gotenv v1.4.2/go.mod h1:ayKnFf/c6rvx/2iiLrJUk1e6plDbT3edrFNGqEflhK0=
github.com/sykesm/zap-logfmt v0.0.4 h1:U2WzRvmIWG1wDLCFY3sz8UeEmsdHQjHFNlIdmroVFaI=
github.com/sykesm/zap-logfmt v0.0.4/go.mod h1:AuBd9xQjAe3URrWT1BBDk2v2onAZHkZkWRMiYZXiZWA=
-github.com/sylabs/sif/v2 v2.9.0 h1:q9K92j1QW4/QLOtKh9YZpJHrXav6x15AVhQGPVLcg+4=
-github.com/sylabs/sif/v2 v2.9.0/go.mod h1:bRdFzcqif0eDjwx0isG4cgTFoKTQn/vfBXVSoP2rB2Y=
-github.com/syndtr/gocapability v0.0.0-20170704070218-db04d3cc01c8/go.mod h1:hkRG7XYTFWNJGYcbNJQlaLq0fg1yr4J4t/NcTQtrfww=
-github.com/syndtr/gocapability v0.0.0-20180916011248-d98352740cb2/go.mod h1:hkRG7XYTFWNJGYcbNJQlaLq0fg1yr4J4t/NcTQtrfww=
+github.com/sylabs/sif/v2 v2.11.5 h1:7ssPH3epSonsTrzbS1YxeJ9KuqAN7ISlSM61a7j/mQM=
+github.com/sylabs/sif/v2 v2.11.5/go.mod h1:GBoZs9LU3e4yJH1dcZ3Akf/jsqYgy5SeguJQC+zd75Y=
github.com/syndtr/gocapability v0.0.0-20200815063812-42c35b437635 h1:kdXcSzyDtseVEc4yCz2qF8ZrQvIDBJLl4S1c3GCXmoI=
github.com/syndtr/gocapability v0.0.0-20200815063812-42c35b437635/go.mod h1:hkRG7XYTFWNJGYcbNJQlaLq0fg1yr4J4t/NcTQtrfww=
-github.com/tchap/go-patricia v2.2.6+incompatible/go.mod h1:bmLyhP68RS6kStMGxByiQ23RP/odRBOTVjwp2cDyi6I=
-github.com/tchap/go-patricia v2.3.0+incompatible h1:GkY4dP3cEfEASBPPkWd+AmjYxhmDkqO9/zg7R0lSQRs=
-github.com/tchap/go-patricia v2.3.0+incompatible/go.mod h1:bmLyhP68RS6kStMGxByiQ23RP/odRBOTVjwp2cDyi6I=
+github.com/tchap/go-patricia/v2 v2.3.1 h1:6rQp39lgIYZ+MHmdEq4xzuk1t7OdC35z/xm0BGhTkes=
+github.com/tchap/go-patricia/v2 v2.3.1/go.mod h1:VZRHKAb53DLaG+nA9EaYYiaEx6YztwDlLElMsnSHD4k=
github.com/theupdateframework/go-tuf v0.5.2 h1:habfDzTmpbzBLIFGWa2ZpVhYvFBoK0C1onC3a4zuPRA=
github.com/theupdateframework/go-tuf v0.5.2/go.mod h1:SyMV5kg5n4uEclsyxXJZI2UxPFJNDc4Y+r7wv+MlvTA=
github.com/theupdateframework/notary v0.7.0 h1:QyagRZ7wlSpjT5N2qQAh/pN+DVqgekv4DzbAiAiEL3c=
@@ -1919,31 +1696,19 @@ github.com/tklauser/go-sysconf v0.3.11 h1:89WgdJhk5SNwJfu+GKyYveZ4IaJ7xAkecBo+Kd
github.com/tklauser/go-sysconf v0.3.11/go.mod h1:GqXfhXY3kiPa0nAXPDIQIWzJbMCB7AmcWpGR8lSZfqI=
github.com/tklauser/numcpus v0.6.0 h1:kebhY2Qt+3U6RNK7UqpYNA+tJ23IBEGKkB7JQBfDYms=
github.com/tklauser/numcpus v0.6.0/go.mod h1:FEZLMke0lhOUG6w2JadTzp0a+Nl8PF/GFkQ5UVIcaL4=
-github.com/tmc/grpc-websocket-proxy v0.0.0-20170815181823-89b8d40f7ca8/go.mod h1:ncp9v5uamzpCO7NfCPTXjqaC+bZgJeR0sMTm6dMHP7U=
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-20201229170055-e5319fda7802 h1:uruHq4dN7GR16kFc5fp3d1RIYzJW5onx8Ybykw2YQFA=
-github.com/tmc/grpc-websocket-proxy v0.0.0-20201229170055-e5319fda7802/go.mod h1:ncp9v5uamzpCO7NfCPTXjqaC+bZgJeR0sMTm6dMHP7U=
-github.com/transparency-dev/merkle v0.0.2 h1:Q9nBoQcZcgPamMkGn7ghV8XiTZ/kRxn1yCG81+twTK4=
+github.com/tmc/grpc-websocket-proxy v0.0.0-20220101234140-673ab2c3ae75 h1:6fotK7otjonDflCTK0BCfls4SPy3NcCVb5dqqmbRknE=
+github.com/tmc/grpc-websocket-proxy v0.0.0-20220101234140-673ab2c3ae75/go.mod h1:KO6IkyS8Y3j8OdNO85qEYBsRPuteD+YciPomcXdrMnk=
github.com/ugorji/go v1.1.4/go.mod h1:uQMGLiO92mf5W77hV/PUCpI3pbzQx3CRekS0kk+RGrc=
github.com/ulikunitz/xz v0.5.10/go.mod h1:nbz6k7qbPmH4IRqmfOplQw/tblSgqTqBwxkY0oWt/14=
github.com/ulikunitz/xz v0.5.11 h1:kpFauv27b6ynzBNT/Xy+1k+fK4WswhN/6PN5WhFAGw8=
github.com/ulikunitz/xz v0.5.11/go.mod h1:nbz6k7qbPmH4IRqmfOplQw/tblSgqTqBwxkY0oWt/14=
-github.com/urfave/cli v0.0.0-20171014202726-7bc6a0acffa5/go.mod h1:70zkFmudgCuE/ngEzBv17Jvp/497gISqfk5gWijbERA=
-github.com/urfave/cli v1.20.0/go.mod h1:70zkFmudgCuE/ngEzBv17Jvp/497gISqfk5gWijbERA=
-github.com/urfave/cli v1.22.1/go.mod h1:Gos4lmkARVdJ6EkW0WaNv/tZAAMe9V7XWyB60NtXRu0=
github.com/urfave/cli v1.22.2/go.mod h1:Gos4lmkARVdJ6EkW0WaNv/tZAAMe9V7XWyB60NtXRu0=
-github.com/urfave/cli v1.22.4/go.mod h1:Gos4lmkARVdJ6EkW0WaNv/tZAAMe9V7XWyB60NtXRu0=
-github.com/urfave/cli v1.22.7 h1:aXiFAgRugfJ27UFDsGJ9DB2FvTC73hlVXFSqq5bo9eU=
-github.com/vbatts/tar-split v0.11.2 h1:Via6XqJr0hceW4wff3QRzD5gAk/tatMw/4ZA7cTlIME=
-github.com/vbatts/tar-split v0.11.2/go.mod h1:vV3ZuO2yWSVsz+pfFzDG/upWH1JhjOiEaWq6kXyQ3VI=
-github.com/vbauerster/mpb/v7 v7.5.3 h1:BkGfmb6nMrrBQDFECR/Q7RkKCw7ylMetCb4079CGs4w=
-github.com/vbauerster/mpb/v7 v7.5.3/go.mod h1:i+h4QY6lmLvBNK2ah1fSreiw3ajskRlBp9AhY/PnuOE=
-github.com/vishvananda/netlink v0.0.0-20181108222139-023a6dafdcdf/go.mod h1:+SR5DhBJrl6ZM7CoCKvpw5BKroDKQ+PJqOg65H/2ktk=
-github.com/vishvananda/netlink v1.1.0/go.mod h1:cTgwzPIzzgDAYoQrMm0EdrjRUBkTqKYppBueQtXaqoE=
-github.com/vishvananda/netlink v1.1.1-0.20201029203352-d40f9887b852/go.mod h1:twkDnbuQxJYemMlGd4JFIcuhgX83tXhKS2B/PRMpOho=
-github.com/vishvananda/netns v0.0.0-20180720170159-13995c7128cc/go.mod h1:ZjcWmFBXmLKZu9Nxj3WKYEafiSqer2rnvPr0en9UNpI=
-github.com/vishvananda/netns v0.0.0-20191106174202-0a2b9b5464df/go.mod h1:JP3t17pCcGlemwknint6hfoeCVQrEMVwxRLRjXpq+BU=
-github.com/vishvananda/netns v0.0.0-20200728191858-db3c7e526aae/go.mod h1:DD4vA1DwXk04H54A1oHXtwZmA0grkVMdPxx/VGLCah0=
+github.com/urfave/cli v1.22.12/go.mod h1:sSBEIC79qR6OvcmsD4U3KABeOTxDqQtdDnaFuUN30b8=
+github.com/vbatts/tar-split v0.11.3 h1:hLFqsOLQ1SsppQNTMpkpPXClLDfC2A3Zgy9OUU+RVck=
+github.com/vbatts/tar-split v0.11.3/go.mod h1:9QlHN18E+fEH7RdG+QAJJcuya3rqT7eXSTY7wGrAokY=
+github.com/vbauerster/mpb/v8 v8.4.0 h1:Jq2iNA7T6SydpMVOwaT+2OBWlXS9Th8KEvBqeu5eeTo=
+github.com/vbauerster/mpb/v8 v8.4.0/go.mod h1:vjp3hSTuCtR+x98/+2vW3eZ8XzxvGoP8CPseHMhiPyc=
github.com/vmihailenco/msgpack v3.3.3+incompatible h1:wapg9xDUZDzGCNFlwc5SqI1rvcciqcxEHac4CYj89xI=
github.com/vmihailenco/msgpack v3.3.3+incompatible/go.mod h1:fy3FlTQTDXWkZ7Bh6AcGMlsjHatGryHQYUTf1ShIgkk=
github.com/vmihailenco/msgpack/v4 v4.3.12/go.mod h1:gborTTJjAo/GWTqqRjrLCn9pgNN+NXzzngzBKDPIqw4=
@@ -1953,8 +1718,6 @@ github.com/vmihailenco/tagparser v0.1.1/go.mod h1:OeAg3pn3UbLjkWt+rN9oFYB6u/cQgq
github.com/vmihailenco/tagparser/v2 v2.0.0 h1:y09buUbR+b5aycVFQs/g70pqKVZNBmxwAhO7/IwNM9g=
github.com/vmware-tanzu/velero v1.10.1 h1:6WYOolZIygHb8FOZtpp8vCqCuy5Mk3qBF1S65L5cjuo=
github.com/vmware-tanzu/velero v1.10.1/go.mod h1:N0J+j8xGSmanGpy1zCRMH2DMGPpwkUj9EZIUXfOlanY=
-github.com/willf/bitset v1.1.11-0.20200630133818-d5bec3311243/go.mod h1:RjeCKbqT1RxIR/KWY6phxZiaY1IyutSBfGjNPySAYV4=
-github.com/willf/bitset v1.1.11/go.mod h1:83CECat5yLh5zVOf4P1ErAgKA5UDvKtgyUABdr3+MjI=
github.com/xanzy/ssh-agent v0.3.3 h1:+/15pJfg/RsTxqYcX6fHqOXZwwMP+2VyYWJeWM2qQFM=
github.com/xanzy/ssh-agent v0.3.3/go.mod h1:6dzNDKs0J9rVPHPhaGCukekBHKqfl+L3KghI1Bc68Uw=
github.com/xdg-go/pbkdf2 v1.0.0 h1:Su7DPu48wXMwC3bs7MCNG+z4FhcyEuz5dlvchbq0B0c=
@@ -1972,13 +1735,12 @@ github.com/xeipuuv/gojsonpointer v0.0.0-20190905194746-02993c407bfb h1:zGWFAtiMc
github.com/xeipuuv/gojsonpointer v0.0.0-20190905194746-02993c407bfb/go.mod h1:N2zxlSyiKSe5eX1tZViRH5QA0qijqEDrYZiPEAiq3wU=
github.com/xeipuuv/gojsonreference v0.0.0-20180127040603-bd5ef7bd5415 h1:EzJWgHovont7NscjpAxXsDA8S8BMYve8Y5+7cuRE7R0=
github.com/xeipuuv/gojsonreference v0.0.0-20180127040603-bd5ef7bd5415/go.mod h1:GwrjFmJcFw6At/Gs6z4yjiIwzuJ1/+UwLxMQDVQXShQ=
-github.com/xeipuuv/gojsonschema v0.0.0-20180618132009-1d523034197f/go.mod h1:5yf86TLmAcydyeJq5YvxkGPE2fm/u4myDekKRoLuqhs=
github.com/xeipuuv/gojsonschema v1.2.0 h1:LhYJRs+L4fBtjZUfuSZIKGeVu0QRy8e5Xi7D17UxZ74=
github.com/xeipuuv/gojsonschema v1.2.0/go.mod h1:anYRn/JVcOK2ZgGU+IjEV4nwlhoK5sQluxsYJ78Id3Y=
github.com/xiang90/probing v0.0.0-20190116061207-43a291ad63a2 h1:eY9dn8+vbi4tKz5Qo6v2eYzo7kUS51QINcR5jNpbZS8=
github.com/xiang90/probing v0.0.0-20190116061207-43a291ad63a2/go.mod h1:UETIi67q53MR2AWcXfiuqkDkRtnGDLqkBTpCHuJHxtU=
-github.com/xlab/treeprint v1.1.0 h1:G/1DjNkPpfZCFt9CSh6b5/nY4VimlbHF3Rh4obvtzDk=
-github.com/xlab/treeprint v1.1.0/go.mod h1:gj5Gd3gPdKtR1ikdDK6fnFLdmIS0X30kTTuNd/WEJu0=
+github.com/xlab/treeprint v1.2.0 h1:HzHnuAF1plUN2zGlAFHbSQP2qJ0ZAD3XF5XD7OesXRQ=
+github.com/xlab/treeprint v1.2.0/go.mod h1:gj5Gd3gPdKtR1ikdDK6fnFLdmIS0X30kTTuNd/WEJu0=
github.com/xordataexchange/crypt v0.0.3-0.20170626215501-b2862e3d0a77/go.mod h1:aYKd//L2LvnjZzWKhF00oedf4jCCReLcmhLdhm1A27Q=
github.com/youmark/pkcs8 v0.0.0-20181117223130-1be2e3e5546d h1:splanxYIlg+5LfHAM6xpdFEAYOk8iySO56hMFq6uLyA=
github.com/youmark/pkcs8 v0.0.0-20181117223130-1be2e3e5546d/go.mod h1:rHwXgn7JulP+udvsHwJoVG1YGAP6VLg4y9I5dyZdqmA=
@@ -2000,47 +1762,26 @@ github.com/zclconf/go-cty v1.12.1 h1:PcupnljUm9EIvbgSHQnHhUr3fO6oFmkOrvs2BAFNXXY
github.com/zclconf/go-cty v1.12.1/go.mod h1:s9IfD1LK5ccNMSWCVFCE2rJfHiZgi7JijgeWIMfhLvA=
github.com/zclconf/go-cty-debug v0.0.0-20191215020915-b22d67c1ba0b/go.mod h1:ZRKQfBXbGkpdV6QMzT3rU1kSTAnfu1dO8dPKjYprgj8=
github.com/zeebo/blake3 v0.2.3 h1:TFoLXsjeXqRNFxSbk35Dk4YtszE/MQQGK10BH4ptoTg=
-github.com/ziutek/mymysql v1.5.4 h1:GB0qdRGsTwQSBVYuVShFBKaXSnSnYYC2d9knnE1LHFs=
-github.com/ziutek/mymysql v1.5.4/go.mod h1:LMSpPZ6DbqWFxNCHW77HeMg9I646SAhApZ/wKdgO/C0=
go.etcd.io/bbolt v1.3.2/go.mod h1:IbVyRI1SCnLcuJnV2u8VeU0CEYM7e686BmAb1XKL+uU=
-go.etcd.io/bbolt v1.3.3/go.mod h1:IbVyRI1SCnLcuJnV2u8VeU0CEYM7e686BmAb1XKL+uU=
-go.etcd.io/bbolt v1.3.5/go.mod h1:G5EMThwa9y8QZGBClrRx5EY+Yw9kAhnjy3bSjsnlVTQ=
-go.etcd.io/bbolt v1.3.6/go.mod h1:qXsaaIqmgQH0T+OPdb99Bf+PKfBBQVAdyD6TY9G8XM4=
go.etcd.io/bbolt v1.3.7 h1:j+zJOnnEjF/kyHlDDgGnVL/AIqIJPq8UoB2GSNfkUfQ=
go.etcd.io/bbolt v1.3.7/go.mod h1:N9Mkw9X8x5fupy0IKsmuqVtoGDyxsaDlbk4Rd05IAQw=
-go.etcd.io/etcd v0.5.0-alpha.5.0.20200910180754-dd1b699fc489 h1:1JFLBqwIgdyHN1ZtgjTBwO+blA6gVOmZurpiMEsETKo=
-go.etcd.io/etcd v0.5.0-alpha.5.0.20200910180754-dd1b699fc489/go.mod h1:yVHk9ub3CSBatqGNg7GRmsnfLWtoW60w4eDYfh7vHDg=
go.etcd.io/etcd/api/v3 v3.5.0/go.mod h1:cbVKeC6lCfl7j/8jBhAK6aIYO9XOjdptoxU/nLQcPvs=
-go.etcd.io/etcd/api/v3 v3.5.6/go.mod h1:KFtNaxGDw4Yx/BA4iPPwevUTAuqcsPxzyX8PHydchN8=
go.etcd.io/etcd/api/v3 v3.5.9 h1:4wSsluwyTbGGmyjJktOf3wFQoTBIURXHnq9n/G/JQHs=
go.etcd.io/etcd/api/v3 v3.5.9/go.mod h1:uyAal843mC8uUVSLWz6eHa/d971iDGnCRpmKd2Z+X8k=
go.etcd.io/etcd/client/pkg/v3 v3.5.0/go.mod h1:IJHfcCEKxYu1Os13ZdwCwIUTUVGYTSAM3YSwc9/Ac1g=
-go.etcd.io/etcd/client/pkg/v3 v3.5.6/go.mod h1:ggrwbk069qxpKPq8/FKkQ3Xq9y39kbFR4LnKszpRXeQ=
go.etcd.io/etcd/client/pkg/v3 v3.5.9 h1:oidDC4+YEuSIQbsR94rY9gur91UPL6DnxDCIYd2IGsE=
go.etcd.io/etcd/client/pkg/v3 v3.5.9/go.mod h1:y+CzeSmkMpWN2Jyu1npecjB9BBnABxGM4pN8cGuJeL4=
go.etcd.io/etcd/client/v2 v2.305.0/go.mod h1:h9puh54ZTgAKtEbut2oe9P4L/oqKCVB6xsXlzd7alYQ=
-go.etcd.io/etcd/client/v2 v2.305.6/go.mod h1:BHha8XJGe8vCIBfWBpbBLVZ4QjOIlfoouvOwydu63E0=
-go.etcd.io/etcd/client/v2 v2.305.8 h1:IGp9Ozt8awy3qRTXSIYJd/o/cr4oUyrm9MF1RJ2dr/c=
-go.etcd.io/etcd/client/v2 v2.305.8/go.mod h1:ZlAsxDK5/10I6xVHhFo9zinCMr/DDLKFetDDXlzKwqE=
-go.etcd.io/etcd/client/v3 v3.5.6/go.mod h1:f6GRinRMCsFVv9Ht42EyY7nfsVGwrNO0WEoS2pRKzQk=
+go.etcd.io/etcd/client/v2 v2.305.9 h1:YZ2OLi0OvR0H75AcgSUajjd5uqKDKocQUqROTG11jIo=
+go.etcd.io/etcd/client/v2 v2.305.9/go.mod h1:0NBdNx9wbxtEQLwAQtrDHwx58m02vXpDcgSYI2seohQ=
go.etcd.io/etcd/client/v3 v3.5.9 h1:r5xghnU7CwbUxD/fbUtRyJGaYNfDun8sp/gTr1hew6E=
go.etcd.io/etcd/client/v3 v3.5.9/go.mod h1:i/Eo5LrZ5IKqpbtpPDuaUnDOUv471oDg8cjQaUr2MbA=
-go.etcd.io/etcd/etcdctl/v3 v3.5.6 h1:HKJrMrJPhol/iM+JpUYLTpq5wIxCy19tGucQTtZ/2KA=
-go.etcd.io/etcd/etcdctl/v3 v3.5.6/go.mod h1:w+nFnWbv2wlCwREmcw7y7UxvaP3H10eWSJeAkl6i860=
-go.etcd.io/etcd/etcdutl/v3 v3.5.6 h1:Pdl25eFdeDfwutUUs13vXJKTqLbLymUpCCr+TWmCmd8=
-go.etcd.io/etcd/etcdutl/v3 v3.5.6/go.mod h1:UnDF5aAyVxEP/VwiW0glKQ79E9/1uG4/NBVN4RLZ8v0=
-go.etcd.io/etcd/pkg/v3 v3.5.6/go.mod h1:qATwUzDb6MLyGWq2nUj+jwXqZJcxkCuabh0P7Cuff3k=
-go.etcd.io/etcd/pkg/v3 v3.5.8 h1:hz6w5Cb4p7dbt642m8Y35Ts9yWPWUCymc3v4Z/aiGEU=
-go.etcd.io/etcd/pkg/v3 v3.5.8/go.mod h1:C17MJkZHJIyJV+wWWx6Jz6YS6BfdkOnUkSwT9uuEO7s=
-go.etcd.io/etcd/raft/v3 v3.5.6/go.mod h1:wL8kkRGx1Hp8FmZUuHfL3K2/OaGIDaXGr1N7i2G07J0=
-go.etcd.io/etcd/raft/v3 v3.5.8 h1:wM4IAfiY1+vrCAkUicIOzkyjpV9MawnAul2KvxeMgy4=
-go.etcd.io/etcd/raft/v3 v3.5.8/go.mod h1:W6P5WxtOMfYNdLSEJX3vc8Pg6LOt+ewI9UCFKcnIexA=
-go.etcd.io/etcd/server/v3 v3.5.6 h1:RXuwaB8AMiV62TqcqIt4O4bG8NWjsxOkDJVT3MZI5Ds=
-go.etcd.io/etcd/server/v3 v3.5.6/go.mod h1:6/Gfe8XTGXQJgLYQ65oGKMfPivb2EASLUSMSWN9Sroo=
-go.etcd.io/etcd/tests/v3 v3.5.6 h1:0akA+RdHfyRI9rRjAzoDo/VAK6Glon8RL/BdqdfHAqE=
-go.etcd.io/etcd/tests/v3 v3.5.6/go.mod h1:jnYOS8eXVQnhxDIWX2YxzMG9SDkuDl6UHUm+mAKgxqg=
-go.etcd.io/etcd/v3 v3.5.6 h1:Oct/Kxr9yfwZaMqD66euFBhr6KaLQYrzae5X1iQdXSA=
-go.etcd.io/etcd/v3 v3.5.6/go.mod h1:xygb0bSpV6OprnKbOYLsctrbw9o4lKsgy2dCMUC4zVo=
+go.etcd.io/etcd/pkg/v3 v3.5.9 h1:6R2jg/aWd/zB9+9JxmijDKStGJAPFsX3e6BeJkMi6eQ=
+go.etcd.io/etcd/pkg/v3 v3.5.9/go.mod h1:BZl0SAShQFk0IpLWR78T/+pyt8AruMHhTNNX73hkNVY=
+go.etcd.io/etcd/raft/v3 v3.5.9 h1:ZZ1GIHoUlHsn0QVqiRysAm3/81Xx7+i2d7nSdWxlOiI=
+go.etcd.io/etcd/raft/v3 v3.5.9/go.mod h1:WnFkqzFdZua4LVlVXQEGhmooLeyS7mqzS4Pf4BCVqXg=
+go.etcd.io/etcd/server/v3 v3.5.9 h1:vomEmmxeztLtS5OEH7d0hBAg4cjVIu9wXuNzUZx2ZA0=
+go.etcd.io/etcd/server/v3 v3.5.9/go.mod h1:GgI1fQClQCFIzuVjlvdbMxNbnISt90gdfYyqiAIt65g=
go.mongodb.org/mongo-driver v1.7.3/go.mod h1:NqaYOwnXWr5Pm7AOpO5QFxKJ503nbMse/R79oO62zWg=
go.mongodb.org/mongo-driver v1.7.5/go.mod h1:VXEWRZ6URJIkUq2SCAyapmhH0ZLRBP+FT4xhp5Zvxng=
go.mongodb.org/mongo-driver v1.10.0/go.mod h1:wsihk0Kdgv8Kqu1Anit4sfK+22vSFbUrAVEYRhCXrA8=
@@ -2058,34 +1799,28 @@ go.opencensus.io v0.22.5/go.mod h1:5pWMHQbX5EPX2/62yrJeAkowc+lfs/XD7Uxpq3pI6kk=
go.opencensus.io v0.23.0/go.mod h1:XItmlyltB5F7CS4xOC1DcqMoFqwtC6OG2xF7mCv7P7E=
go.opencensus.io v0.24.0 h1:y73uSU6J157QMP2kn2r30vwW1A2W2WFwSCGnAVxeaD0=
go.opencensus.io v0.24.0/go.mod h1:vNK8G9p7aAivkbmorf4v+7Hgx+Zs0yY+0fOtgBfjQKo=
-go.opentelemetry.io/contrib/instrumentation/google.golang.org/grpc/otelgrpc v0.25.0/go.mod h1:E5NNboN0UqSAki0Atn9kVwaN7I+l25gGxDqBueo/74E=
-go.opentelemetry.io/contrib/instrumentation/google.golang.org/grpc/otelgrpc v0.35.0 h1:xFSRQBbXF6VvYRf2lqMJXxoB72XI1K/azav8TekHHSw=
-go.opentelemetry.io/contrib/instrumentation/google.golang.org/grpc/otelgrpc v0.35.0/go.mod h1:h8TWwRAhQpOd0aM5nYsRD8+flnkj+526GEIVlarH7eY=
-go.opentelemetry.io/otel v1.0.1/go.mod h1:OPEOD4jIT2SlZPMmwT6FqZz2C0ZNdQqiWcoK6M0SNFU=
-go.opentelemetry.io/otel v1.14.0 h1:/79Huy8wbf5DnIPhemGB+zEPVwnN6fuQybr/SRXa6hM=
-go.opentelemetry.io/otel v1.14.0/go.mod h1:o4buv+dJzx8rohcUeRmWUZhqupFvzWis188WlggnNeU=
-go.opentelemetry.io/otel/exporters/otlp/internal/retry v1.11.2 h1:htgM8vZIF8oPSCxa341e3IZ4yr/sKxgu8KZYllByiVY=
-go.opentelemetry.io/otel/exporters/otlp/internal/retry v1.11.2/go.mod h1:rqbht/LlhVBgn5+k3M5QK96K5Xb0DvXpMJ5SFQpY6uw=
-go.opentelemetry.io/otel/exporters/otlp/otlptrace v1.0.1/go.mod h1:Kv8liBeVNFkkkbilbgWRpV+wWuu+H5xdOT6HAgd30iw=
-go.opentelemetry.io/otel/exporters/otlp/otlptrace v1.11.2 h1:fqR1kli93643au1RKo0Uma3d2aPQKT+WBKfTSBaKbOc=
-go.opentelemetry.io/otel/exporters/otlp/otlptrace v1.11.2/go.mod h1:5Qn6qvgkMsLDX+sYK64rHb1FPhpn0UtxF+ouX1uhyJE=
-go.opentelemetry.io/otel/exporters/otlp/otlptrace/otlptracegrpc v1.0.1/go.mod h1:xOvWoTOrQjxjW61xtOmD/WKGRYb/P4NzRo3bs65U6Rk=
-go.opentelemetry.io/otel/exporters/otlp/otlptrace/otlptracegrpc v1.11.2 h1:ERwKPn9Aer7Gxsc0+ZlutlH1bEEAUXAUhqm3Y45ABbk=
-go.opentelemetry.io/otel/exporters/otlp/otlptrace/otlptracegrpc v1.11.2/go.mod h1:jWZUM2MWhWCJ9J9xVbRx7tzK1mXKpAlze4CeulycwVY=
-go.opentelemetry.io/otel/sdk v1.0.1/go.mod h1:HrdXne+BiwsOHYYkBE5ysIcv2bvdZstxzmCQhxTcZkI=
-go.opentelemetry.io/otel/sdk v1.14.0 h1:PDCppFRDq8A1jL9v6KMI6dYesaq+DFcDZvjsoGvxGzY=
-go.opentelemetry.io/otel/sdk v1.14.0/go.mod h1:bwIC5TjrNG6QDCHNWvW4HLHtUQ4I+VQDsnjhvyZCALM=
-go.opentelemetry.io/otel/trace v1.0.1/go.mod h1:5g4i4fKLaX2BQpSBsxw8YYcgKpMMSW3x7ZTuYBr3sUk=
-go.opentelemetry.io/otel/trace v1.14.0 h1:wp2Mmvj41tDsyAJXiWDWpfNsOiIyd38fy85pyKcFq/M=
-go.opentelemetry.io/otel/trace v1.14.0/go.mod h1:8avnQLK+CG77yNLUae4ea2JDQ6iT+gozhnZjy/rw9G8=
+go.opentelemetry.io/contrib/instrumentation/google.golang.org/grpc/otelgrpc v0.40.0 h1:5jD3teb4Qh7mx/nfzq4jO2WFFpvXD0vYWFDrdvNWmXk=
+go.opentelemetry.io/contrib/instrumentation/google.golang.org/grpc/otelgrpc v0.40.0/go.mod h1:UMklln0+MRhZC4e3PwmN3pCtq4DyIadWw4yikh6bNrw=
+go.opentelemetry.io/otel v1.15.0 h1:NIl24d4eiLJPM0vKn4HjLYM+UZf6gSfi9Z+NmCxkWbk=
+go.opentelemetry.io/otel v1.15.0/go.mod h1:qfwLEbWhLPk5gyWrne4XnF0lC8wtywbuJbgfAE3zbek=
+go.opentelemetry.io/otel/exporters/otlp/internal/retry v1.15.0 h1:ZSdnH1x5Gm/eUFNQquwSt4/LMCOqS6KPlI9qaTKx5Ho=
+go.opentelemetry.io/otel/exporters/otlp/internal/retry v1.15.0/go.mod h1:uOTV75+LOzV+ODmL8ahRLWkFA3eQcSC2aAsbxIu4duk=
+go.opentelemetry.io/otel/exporters/otlp/otlptrace v1.15.0 h1:rk5I7PaOk5NGQHfHR2Rz6MgdA8AYQSHwsigFsOxEC1c=
+go.opentelemetry.io/otel/exporters/otlp/otlptrace v1.15.0/go.mod h1:pvkFJxNUXyJ5i8u6m8NIcqkoOf/65VM2mSyBbBJfeVQ=
+go.opentelemetry.io/otel/exporters/otlp/otlptrace/otlptracegrpc v1.15.0 h1:rHD0vfQbtki6/FnsMzTpAOgdv+Ku+T6R47MZXmgelf8=
+go.opentelemetry.io/otel/exporters/otlp/otlptrace/otlptracegrpc v1.15.0/go.mod h1:RPagkaZrpwD+rSwQjzos6rBLsHOvenOqufCj4/7I46E=
+go.opentelemetry.io/otel/metric v0.37.0 h1:pHDQuLQOZwYD+Km0eb657A25NaRzy0a+eLyKfDXedEs=
+go.opentelemetry.io/otel/metric v0.37.0/go.mod h1:DmdaHfGt54iV6UKxsV9slj2bBRJcKC1B1uvDLIioc1s=
+go.opentelemetry.io/otel/sdk v1.15.0 h1:jZTCkRRd08nxD6w7rIaZeDNGZGGQstH3SfLQ3ZsKICk=
+go.opentelemetry.io/otel/sdk v1.15.0/go.mod h1:XDEMrYWzJ4YlC17i6Luih2lwDw2j6G0PkUfr1ZqE+rQ=
+go.opentelemetry.io/otel/trace v1.15.0 h1:5Fwje4O2ooOxkfyqI/kJwxWotggDLix4BSAvpE1wlpo=
+go.opentelemetry.io/otel/trace v1.15.0/go.mod h1:CUsmE2Ht1CRkvE8OsMESvraoZrrcgD1J2W8GV1ev0Y4=
go.opentelemetry.io/proto/otlp v0.7.0/go.mod h1:PqfVotwruBrMGOCsRd/89rSnXhoiJIqeYNgFYFoEGnI=
-go.opentelemetry.io/proto/otlp v0.9.0/go.mod h1:1vKfU9rv61e9EVGthD1zNvUbiwPcimSsOPU9brfSHJg=
go.opentelemetry.io/proto/otlp v0.15.0/go.mod h1:H7XAot3MsfNsj7EXtrA2q5xSNQ10UqI405h3+duxN4U=
go.opentelemetry.io/proto/otlp v0.19.0 h1:IVN6GR+mhC4s5yfcTbmzHYODqvWAp3ZedA2SJPI1Nnw=
go.opentelemetry.io/proto/otlp v0.19.0/go.mod h1:H7XAot3MsfNsj7EXtrA2q5xSNQ10UqI405h3+duxN4U=
-go.starlark.net v0.0.0-20201006213952-227f4aabceb5 h1:ApvY/1gw+Yiqb/FKeks3KnVPWpkR3xzij82XPKLjJVw=
-go.starlark.net v0.0.0-20201006213952-227f4aabceb5/go.mod h1:f0znQkUKRrkk36XxWbGjMqQM8wGv/xHBVE2qc3B5oFU=
-go.uber.org/atomic v1.3.2/go.mod h1:gD2HeocX3+yG+ygLZcrzQJaqmWj9AIm7n08wl/qW/PE=
+go.starlark.net v0.0.0-20230525235612-a134d8f9ddca h1:VdD38733bfYv5tUZwEIskMM93VanwNIi5bIKnDrJdEY=
+go.starlark.net v0.0.0-20230525235612-a134d8f9ddca/go.mod h1:jxU+3+j+71eXOW14274+SmmuW82qJzl6iZSeqEtTGds=
go.uber.org/atomic v1.4.0/go.mod h1:gD2HeocX3+yG+ygLZcrzQJaqmWj9AIm7n08wl/qW/PE=
go.uber.org/atomic v1.5.0/go.mod h1:sABNBOSYdrvTF6hTgEIbc7YasKWGhgEQZyfxyTvoXHQ=
go.uber.org/atomic v1.7.0/go.mod h1:fEN4uk6kAWBTFdckzkM89CLk9XfWZrxpCo0nPH17wJc=
@@ -2099,8 +1834,8 @@ go.uber.org/goleak v1.2.1 h1:NBol2c7O1ZokfZ0LEU9K6Whx/KnwvepVetCUhtKja4A=
go.uber.org/multierr v1.1.0/go.mod h1:wR5kodmAFQ0UK8QlbwjlSNy0Z68gJhDJUG5sjR94q/0=
go.uber.org/multierr v1.3.0/go.mod h1:VgVr7evmIr6uPjLBxg28wmKNXyqE9akIJ5XnfpiKl+4=
go.uber.org/multierr v1.6.0/go.mod h1:cdWPpRnG4AhwMwsgIHip0KRBQjJy5kYEpYjJxpXp9iU=
-go.uber.org/multierr v1.9.0 h1:7fIwc/ZtS0q++VgcfqFDxSBZVv/Xo49/SYnDFupUwlI=
-go.uber.org/multierr v1.9.0/go.mod h1:X2jQV1h+kxSjClGpnseKVIxpmcjrj7MNnI0bnlfKTVQ=
+go.uber.org/multierr v1.11.0 h1:blXXJkSxSSfBVBlC76pxqeO+LN3aDfLQo+309xJstO0=
+go.uber.org/multierr v1.11.0/go.mod h1:20+QtiLqy0Nd6FdQB9TLXag12DsQkrbs3htMFfDN80Y=
go.uber.org/tools v0.0.0-20190618225709-2cfd321de3ee/go.mod h1:vJERXedbb3MVM5f9Ejo0C68/HhF8uaILCdgjnY+goOA=
go.uber.org/zap v1.10.0/go.mod h1:vwi/ZaCAaUcBkycHslxD9B2zi4UTXhF60s6SWpuDF0Q=
go.uber.org/zap v1.12.0/go.mod h1:zwrFLgMcdUuIBviXEYEH1YKNaOBnKXsx2IPda5bBwHM=
@@ -2108,16 +1843,10 @@ go.uber.org/zap v1.17.0/go.mod h1:MXVU+bhUf/A7Xi2HNOnopQOrmycQ5Ih87HtOu4q5SSo=
go.uber.org/zap v1.18.1/go.mod h1:xg/QME4nWcxGxrpdeYfq7UvYrLh66cuVKdrbD1XF/NI=
go.uber.org/zap v1.24.0 h1:FiJd5l1UOLj0wCgbSE0rwwXHzEdAZS6hiiSnxJN/D60=
go.uber.org/zap v1.24.0/go.mod h1:2kMP+WWQ8aoFoedH3T2sq6iJ2yDWpHbP0f6MQbS9Gkg=
-go4.org/intern v0.0.0-20211027215823-ae77deb06f29 h1:UXLjNohABv4S58tHmeuIZDO6e3mHpW2Dx33gaNt03LE=
-go4.org/intern v0.0.0-20211027215823-ae77deb06f29/go.mod h1:cS2ma+47FKrLPdXFpr7CuxiTW3eyJbWew4qx0qtQWDA=
-go4.org/unsafe/assume-no-moving-gc v0.0.0-20211027215541-db492cf91b37/go.mod h1:FftLjUGFEDu5k8lt0ddY+HcrH/qU/0qk+H8j9/nTl3E=
-go4.org/unsafe/assume-no-moving-gc v0.0.0-20220617031537-928513b29760/go.mod h1:FftLjUGFEDu5k8lt0ddY+HcrH/qU/0qk+H8j9/nTl3E=
-go4.org/unsafe/assume-no-moving-gc v0.0.0-20230221090011-e4bae7ad2296 h1:QJ/xcIANMLApehfgPCHnfK1hZiaMmbaTVmPv7DAoTbo=
-go4.org/unsafe/assume-no-moving-gc v0.0.0-20230221090011-e4bae7ad2296/go.mod h1:FftLjUGFEDu5k8lt0ddY+HcrH/qU/0qk+H8j9/nTl3E=
+go4.org/netipx v0.0.0-20230728184502-ec4c8b891b28 h1:zLxFnORHDFTSkJPawMU7LzsuGQJ4MUFS653jJHpORow=
+go4.org/netipx v0.0.0-20230728184502-ec4c8b891b28/go.mod h1:TQvodOM+hJTioNQJilmLXu08JNb8i+ccq418+KWu1/Y=
golang.org/x/arch v0.1.0/go.mod h1:5om86z9Hs0C8fWVUuoMHwpExlXzs5Tkyp9hOrfG7pp8=
-golang.org/x/crypto v0.0.0-20171113213409-9f005a07e0d3/go.mod h1:6SG95UA2DQfeDnfUPMdvaQW0Q7yPrPDi9nlGo2tz2b4=
golang.org/x/crypto v0.0.0-20180904163835-0709b304e793/go.mod h1:6SG95UA2DQfeDnfUPMdvaQW0Q7yPrPDi9nlGo2tz2b4=
-golang.org/x/crypto v0.0.0-20181009213950-7c1a557ab941/go.mod h1:6SG95UA2DQfeDnfUPMdvaQW0Q7yPrPDi9nlGo2tz2b4=
golang.org/x/crypto v0.0.0-20181029021203-45a5f77698d3/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-20190325154230-a5d413f7728c/go.mod h1:djNgcEr1/C05ACkg1iLfiJU5Ep61QUkGW8qpdssI0+w=
@@ -2125,31 +1854,30 @@ golang.org/x/crypto v0.0.0-20190422162423-af44ce270edf/go.mod h1:WFFai1msRO1wXaE
golang.org/x/crypto v0.0.0-20190510104115-cbcb75029529/go.mod h1:yigFU9vqHzYiE8UmvKecakEJjdnWj3jj499lnFckfCI=
golang.org/x/crypto v0.0.0-20190605123033-f99c8df09eb5/go.mod h1:yigFU9vqHzYiE8UmvKecakEJjdnWj3jj499lnFckfCI=
golang.org/x/crypto v0.0.0-20190611184440-5c40567a22f8/go.mod h1:yigFU9vqHzYiE8UmvKecakEJjdnWj3jj499lnFckfCI=
-golang.org/x/crypto v0.0.0-20190701094942-4def268fd1a4/go.mod h1:yigFU9vqHzYiE8UmvKecakEJjdnWj3jj499lnFckfCI=
golang.org/x/crypto v0.0.0-20191011191535-87dc89f01550/go.mod h1:yigFU9vqHzYiE8UmvKecakEJjdnWj3jj499lnFckfCI=
golang.org/x/crypto v0.0.0-20191206172530-e9b2fee46413/go.mod h1:LzIPMQfyMNhhGPhUkYOs5KpL4U8rLKemX1yGLhDgUto=
golang.org/x/crypto v0.0.0-20200302210943-78000ba7a073/go.mod h1:LzIPMQfyMNhhGPhUkYOs5KpL4U8rLKemX1yGLhDgUto=
golang.org/x/crypto v0.0.0-20200414173820-0848c9571904/go.mod h1:LzIPMQfyMNhhGPhUkYOs5KpL4U8rLKemX1yGLhDgUto=
golang.org/x/crypto v0.0.0-20200622213623-75b288015ac9/go.mod h1:LzIPMQfyMNhhGPhUkYOs5KpL4U8rLKemX1yGLhDgUto=
-golang.org/x/crypto v0.0.0-20200728195943-123391ffb6de/go.mod h1:LzIPMQfyMNhhGPhUkYOs5KpL4U8rLKemX1yGLhDgUto=
golang.org/x/crypto v0.0.0-20200820211705-5c72a883971a/go.mod h1:LzIPMQfyMNhhGPhUkYOs5KpL4U8rLKemX1yGLhDgUto=
-golang.org/x/crypto v0.0.0-20201002170205-7f63de1d35b0/go.mod h1:LzIPMQfyMNhhGPhUkYOs5KpL4U8rLKemX1yGLhDgUto=
golang.org/x/crypto v0.0.0-20201117144127-c1f2f97bffc9/go.mod h1:jdWPYTVW3xRLrWPugEBEK3UY2ZEsg3UU495nc5E+M+I=
golang.org/x/crypto v0.0.0-20201221181555-eec23a3978ad/go.mod h1:jdWPYTVW3xRLrWPugEBEK3UY2ZEsg3UU495nc5E+M+I=
golang.org/x/crypto v0.0.0-20210322153248-0c34fe9e7dc2/go.mod h1:T9bdIzuCu7OtxOm1hfPfRQxPLYneinmdGuTeoZ9dtd4=
golang.org/x/crypto v0.0.0-20210921155107-089bfa567519/go.mod h1:GvvjBRRGRdwPK5ydBHafDWAxML/pGHZbMvKqRZ5+Abc=
golang.org/x/crypto v0.0.0-20211215153901-e495a2d5b3d3/go.mod h1:IxCIyHEi3zRg3s0A5j5BB6A9Jmi73HwBIUl50j+osU4=
golang.org/x/crypto v0.0.0-20220314234659-1baeb1ce4c0b/go.mod h1:IxCIyHEi3zRg3s0A5j5BB6A9Jmi73HwBIUl50j+osU4=
-golang.org/x/crypto v0.0.0-20220411220226-7b82a4e95df4/go.mod h1:IxCIyHEi3zRg3s0A5j5BB6A9Jmi73HwBIUl50j+osU4=
golang.org/x/crypto v0.0.0-20220525230936-793ad666bf5e/go.mod h1:IxCIyHEi3zRg3s0A5j5BB6A9Jmi73HwBIUl50j+osU4=
golang.org/x/crypto v0.0.0-20220622213112-05595931fe9d/go.mod h1:IxCIyHEi3zRg3s0A5j5BB6A9Jmi73HwBIUl50j+osU4=
golang.org/x/crypto v0.0.0-20220722155217-630584e8d5aa/go.mod h1:IxCIyHEi3zRg3s0A5j5BB6A9Jmi73HwBIUl50j+osU4=
golang.org/x/crypto v0.0.0-20220826181053-bd7e27e6170d/go.mod h1:IxCIyHEi3zRg3s0A5j5BB6A9Jmi73HwBIUl50j+osU4=
golang.org/x/crypto v0.1.0/go.mod h1:RecgLatLF4+eUMCP1PoPZQb+cVrJcOPbHkTkbkB9sbw=
golang.org/x/crypto v0.3.0/go.mod h1:hebNnKkNXi2UzZN1eVRvBB7co0a+JxK6XbPiWVs/3J4=
+golang.org/x/crypto v0.3.1-0.20221117191849-2c476679df9a/go.mod h1:hebNnKkNXi2UzZN1eVRvBB7co0a+JxK6XbPiWVs/3J4=
+golang.org/x/crypto v0.5.0/go.mod h1:NK/OQwhpMQP3MwtdjgLlYHnH9ebylxKWv3e0fK+mkQU=
golang.org/x/crypto v0.6.0/go.mod h1:OFC/31mSvZgRz0V1QTNCzfAI1aIRzbiufJtkMIlEp58=
-golang.org/x/crypto v0.12.0 h1:tFM/ta59kqch6LlvYnPa0yx5a83cL2nHflFhYKvv9Yk=
-golang.org/x/crypto v0.12.0/go.mod h1:NF0Gs7EO5K4qLn+Ylc+fih8BSTeIjAP05siRnAh98yw=
+golang.org/x/crypto v0.7.0/go.mod h1:pYwdfH91IfpZVANVyUOhSIPZaFoJGxTFbZhFTx+dXZU=
+golang.org/x/crypto v0.13.0 h1:mvySKfSWJ+UKUii46M40LOvyWfN0s2U+46/jDd0e6Ck=
+golang.org/x/crypto v0.13.0/go.mod h1:y6Z2r+Rw4iayiXXAIxJIDAJ1zMW4yaTpebo8fPOliYc=
golang.org/x/exp v0.0.0-20190121172915-509febef88a4/go.mod h1:CJ0aWSM057203Lf6IL+f9T1iT9GByDxfZKAQTCR3kQA=
golang.org/x/exp v0.0.0-20190306152737-a1d7652674e8/go.mod h1:CJ0aWSM057203Lf6IL+f9T1iT9GByDxfZKAQTCR3kQA=
golang.org/x/exp v0.0.0-20190510132918-efd6b22b2522/go.mod h1:ZjyILWgesfNpC6sMxTJOJm9Kp84zZh5NQWvqDGG3Qr8=
@@ -2190,12 +1918,12 @@ golang.org/x/mod v0.5.0/go.mod h1:5OXOZSfqPIIbmVBIIKWRFfZjPR0E5r58TLhUjH0a2Ro=
golang.org/x/mod v0.6.0-dev.0.20220419223038-86c51ed26bb4/go.mod h1:jJ57K6gSWd91VN4djpZkiMVwK6gcyfeH4XE8wZrZaV4=
golang.org/x/mod v0.6.0/go.mod h1:4mET923SAdbXp2ki8ey+zGs1SLqsuM2Y0uvdZR/fUNI=
golang.org/x/mod v0.7.0/go.mod h1:iBbtSCu2XBx23ZKBPSOrRkjjQPZFPuis4dIYUhu/chs=
+golang.org/x/mod v0.8.0/go.mod h1:iBbtSCu2XBx23ZKBPSOrRkjjQPZFPuis4dIYUhu/chs=
golang.org/x/mod v0.11.0 h1:bUO06HqtnRcc/7l71XBe4WcqTZ+3AH1J59zWDDwLKgU=
golang.org/x/mod v0.11.0/go.mod h1:iBbtSCu2XBx23ZKBPSOrRkjjQPZFPuis4dIYUhu/chs=
golang.org/x/net v0.0.0-20180724234803-3673e40ba225/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4=
golang.org/x/net v0.0.0-20180811021610-c39426892332/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-20181011144130-49bb7cea24b1/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4=
golang.org/x/net v0.0.0-20181023162649-9b4f9f5ad519/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-20181201002055-351d144fa1fc/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4=
@@ -2212,7 +1940,6 @@ golang.org/x/net v0.0.0-20190613194153-d28f0bde5980/go.mod h1:z5CRVTTTmAJ677TzLL
golang.org/x/net v0.0.0-20190620200207-3b0461eec859/go.mod h1:z5CRVTTTmAJ677TzLLGU+0bjPO0LkuOLi4/5GtJWs/s=
golang.org/x/net v0.0.0-20190628185345-da137c7871d7/go.mod h1:z5CRVTTTmAJ677TzLLGU+0bjPO0LkuOLi4/5GtJWs/s=
golang.org/x/net v0.0.0-20190724013045-ca1201d0de80/go.mod h1:z5CRVTTTmAJ677TzLLGU+0bjPO0LkuOLi4/5GtJWs/s=
-golang.org/x/net v0.0.0-20190813141303-74dc4d7220e7/go.mod h1:z5CRVTTTmAJ677TzLLGU+0bjPO0LkuOLi4/5GtJWs/s=
golang.org/x/net v0.0.0-20190827160401-ba9fcec4b297/go.mod h1:z5CRVTTTmAJ677TzLLGU+0bjPO0LkuOLi4/5GtJWs/s=
golang.org/x/net v0.0.0-20190923162816-aa69164e4478/go.mod h1:z5CRVTTTmAJ677TzLLGU+0bjPO0LkuOLi4/5GtJWs/s=
golang.org/x/net v0.0.0-20191004110552-13f9640d40b9/go.mod h1:z5CRVTTTmAJ677TzLLGU+0bjPO0LkuOLi4/5GtJWs/s=
@@ -2230,13 +1957,11 @@ golang.org/x/net v0.0.0-20200520182314-0ba52f642ac2/go.mod h1:qpuaurCH72eLCgpAm/
golang.org/x/net v0.0.0-20200625001655-4c5254603344/go.mod h1:/O7V0waA8r7cgGh81Ro3o1hOxt32SMVPicZroKQ2sZA=
golang.org/x/net v0.0.0-20200707034311-ab3426394381/go.mod h1:/O7V0waA8r7cgGh81Ro3o1hOxt32SMVPicZroKQ2sZA=
golang.org/x/net v0.0.0-20200822124328-c89045814202/go.mod h1:/O7V0waA8r7cgGh81Ro3o1hOxt32SMVPicZroKQ2sZA=
-golang.org/x/net v0.0.0-20201006153459-a7d1128ccaa0/go.mod h1:sp8m0HH+o8qH0wwXwYZr8TS3Oi6o0r6Gce1SSxlDquU=
golang.org/x/net v0.0.0-20201021035429-f5854403a974/go.mod h1:sp8m0HH+o8qH0wwXwYZr8TS3Oi6o0r6Gce1SSxlDquU=
golang.org/x/net v0.0.0-20201031054903-ff519b6c9102/go.mod h1:sp8m0HH+o8qH0wwXwYZr8TS3Oi6o0r6Gce1SSxlDquU=
golang.org/x/net v0.0.0-20201110031124-69a78807bb2b/go.mod h1:sp8m0HH+o8qH0wwXwYZr8TS3Oi6o0r6Gce1SSxlDquU=
golang.org/x/net v0.0.0-20201202161906-c7110b5ffcbb/go.mod h1:sp8m0HH+o8qH0wwXwYZr8TS3Oi6o0r6Gce1SSxlDquU=
golang.org/x/net v0.0.0-20201209123823-ac852fbbde11/go.mod h1:m0MpNAwzfU5UDzcl9v0D8zg8gWTRqZa9RBIspLL5mdg=
-golang.org/x/net v0.0.0-20201224014010-6772e930b67b/go.mod h1:m0MpNAwzfU5UDzcl9v0D8zg8gWTRqZa9RBIspLL5mdg=
golang.org/x/net v0.0.0-20210119194325-5f4716e94777/go.mod h1:m0MpNAwzfU5UDzcl9v0D8zg8gWTRqZa9RBIspLL5mdg=
golang.org/x/net v0.0.0-20210226172049-e18ecbb05110/go.mod h1:m0MpNAwzfU5UDzcl9v0D8zg8gWTRqZa9RBIspLL5mdg=
golang.org/x/net v0.0.0-20210316092652-d523dce5a7f4/go.mod h1:RBQZq4jEuRlivfhVLdyRGr576XBO4/greRjx4P4O3yc=
@@ -2245,8 +1970,8 @@ golang.org/x/net v0.0.0-20210421230115-4e50805a0758/go.mod h1:72T/g9IO56b78aLF+1
golang.org/x/net v0.0.0-20210503060351-7fd8e65b6420/go.mod h1:9nx3DQGgdP8bBQD5qxJ1jj9UTztislL4KSBs9R2vV5Y=
golang.org/x/net v0.0.0-20210805182204-aaa1db679c0d/go.mod h1:9nx3DQGgdP8bBQD5qxJ1jj9UTztislL4KSBs9R2vV5Y=
golang.org/x/net v0.0.0-20210813160813-60bc85c4be6d/go.mod h1:9nx3DQGgdP8bBQD5qxJ1jj9UTztislL4KSBs9R2vV5Y=
-golang.org/x/net v0.0.0-20210825183410-e898025ed96a/go.mod h1:9nx3DQGgdP8bBQD5qxJ1jj9UTztislL4KSBs9R2vV5Y=
golang.org/x/net v0.0.0-20211112202133-69e39bad7dc2/go.mod h1:9nx3DQGgdP8bBQD5qxJ1jj9UTztislL4KSBs9R2vV5Y=
+golang.org/x/net v0.0.0-20211123203042-d83791d6bcd9/go.mod h1:9nx3DQGgdP8bBQD5qxJ1jj9UTztislL4KSBs9R2vV5Y=
golang.org/x/net v0.0.0-20220127200216-cd36cc0744dd/go.mod h1:CfG3xpIq0wQ8r1q4Su4UZFWDARRcnwPjda9FqA0JpMk=
golang.org/x/net v0.0.0-20220225172249-27dd8689420f/go.mod h1:CfG3xpIq0wQ8r1q4Su4UZFWDARRcnwPjda9FqA0JpMk=
golang.org/x/net v0.0.0-20220325170049-de3da57026de/go.mod h1:CfG3xpIq0wQ8r1q4Su4UZFWDARRcnwPjda9FqA0JpMk=
@@ -2266,6 +1991,7 @@ golang.org/x/net v0.2.0/go.mod h1:KqCZLdyyvdV855qA2rE3GC2aiw5xGR5TEjj8smXukLY=
golang.org/x/net v0.5.0/go.mod h1:DivGGAXEgPSlEBzxGzZI+ZLohi+xUj054jfeKui00ws=
golang.org/x/net v0.6.0/go.mod h1:2Tu9+aMcznHK/AK1HMvgo6xiTLG5rD5rZLDS+rp2Bjs=
golang.org/x/net v0.7.0/go.mod h1:2Tu9+aMcznHK/AK1HMvgo6xiTLG5rD5rZLDS+rp2Bjs=
+golang.org/x/net v0.8.0/go.mod h1:QVkue5JL9kW//ek3r6jTKnTFis1tRmNAW2P1shuFdJc=
golang.org/x/net v0.14.0 h1:BONx9s002vGdD9umnlX1Po8vOZmrgH34qlHcD1MfK14=
golang.org/x/net v0.14.0/go.mod h1:PpSgVXXLK0OxS0F31C1/tv6XNguvCrnXIDrFMspZIUI=
golang.org/x/oauth2 v0.0.0-20180821212333-d2e6202438be/go.mod h1:N/0e6XlmueqKjAGxoOufVs8QHGRruUQn6yWY3a++T0U=
@@ -2324,6 +2050,7 @@ golang.org/x/sys v0.0.0-20181107165924-66b7b1311ac8/go.mod h1:STP8DvDyc/dI5b8T5h
golang.org/x/sys v0.0.0-20181116152217-5ac8a444bdc5/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY=
golang.org/x/sys v0.0.0-20181122145206-62eef0e2fa9b/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-20190312061237-fead79001313/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
golang.org/x/sys v0.0.0-20190403152447-81d4e9dc473e/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
golang.org/x/sys v0.0.0-20190412213103-97732733099d/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
@@ -2331,39 +2058,28 @@ golang.org/x/sys v0.0.0-20190419153524-e8e3143a4f4a/go.mod h1:h1NjWce9XRLGQEsW7w
golang.org/x/sys v0.0.0-20190422165155-953cdadca894/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
golang.org/x/sys v0.0.0-20190502145724-3ef323f4f1fd/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
golang.org/x/sys v0.0.0-20190507160741-ecd444e8653b/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
-golang.org/x/sys v0.0.0-20190514135907-3a4b5fb9f71f/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
-golang.org/x/sys v0.0.0-20190522044717-8097e1b27ff5/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
golang.org/x/sys v0.0.0-20190524152521-dbbf3f1254d4/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
golang.org/x/sys v0.0.0-20190531175056-4c3a928424d2/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
golang.org/x/sys v0.0.0-20190606165138-5da285871e9c/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
-golang.org/x/sys v0.0.0-20190606203320-7fc4e5ec1444/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
golang.org/x/sys v0.0.0-20190616124812-15dcb6c0061f/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
golang.org/x/sys v0.0.0-20190624142023-c5567b49c5d0/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
-golang.org/x/sys v0.0.0-20190626221950-04f50cda93cb/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
golang.org/x/sys v0.0.0-20190726091711-fc99dfbffb4e/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
golang.org/x/sys v0.0.0-20190801041406-cbf593c0f2f3/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
-golang.org/x/sys v0.0.0-20190812073006-9eafafc0a87e/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
-golang.org/x/sys v0.0.0-20190826190057-c7b8b68b1456/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
-golang.org/x/sys v0.0.0-20190904154756-749cb33beabd/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
golang.org/x/sys v0.0.0-20190916202348-b4ddaad3f8a3/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
golang.org/x/sys v0.0.0-20191001151750-bb3f8db39f24/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
golang.org/x/sys v0.0.0-20191005200804-aed5e4c7ecf9/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
golang.org/x/sys v0.0.0-20191010194322-b09406accb47/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
golang.org/x/sys v0.0.0-20191022100944-742c48ecaeb7/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
golang.org/x/sys v0.0.0-20191026070338-33540a1f6037/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
-golang.org/x/sys v0.0.0-20191115151921-52ab43148777/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
golang.org/x/sys v0.0.0-20191120155948-bd437916bb0e/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
golang.org/x/sys v0.0.0-20191204072324-ce4227a45e2e/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
-golang.org/x/sys v0.0.0-20191210023423-ac6580df4449/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
golang.org/x/sys v0.0.0-20191228213918-04cbcbbfeed8/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
-golang.org/x/sys v0.0.0-20200106162015-b016eb3dc98e/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
golang.org/x/sys v0.0.0-20200113162924-86b910548bc1/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
+golang.org/x/sys v0.0.0-20200116001909-b77594299b42/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
golang.org/x/sys v0.0.0-20200120151820-655fe14d7479/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
golang.org/x/sys v0.0.0-20200122134326-e047566fdf82/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
-golang.org/x/sys v0.0.0-20200124204421-9fbb57f87de9/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
golang.org/x/sys v0.0.0-20200202164722-d101bd2416d5/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
golang.org/x/sys v0.0.0-20200212091648-12a6c2dcc1e4/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
-golang.org/x/sys v0.0.0-20200217220822-9197077df867/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
golang.org/x/sys v0.0.0-20200223170610-d5e6a3e2c0ae/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
golang.org/x/sys v0.0.0-20200302150141-5c8b2ff67527/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
golang.org/x/sys v0.0.0-20200323222414-85ca7c5b95cd/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
@@ -2372,23 +2088,12 @@ golang.org/x/sys v0.0.0-20200501052902-10377860bb8e/go.mod h1:h1NjWce9XRLGQEsW7w
golang.org/x/sys v0.0.0-20200511232937-7e40ca221e25/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
golang.org/x/sys v0.0.0-20200515095857-1151b9dac4a9/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
golang.org/x/sys v0.0.0-20200523222454-059865788121/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
-golang.org/x/sys v0.0.0-20200615200032-f1bc736245b1/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
golang.org/x/sys v0.0.0-20200622214017-ed371f2e16b4/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
-golang.org/x/sys v0.0.0-20200625212154-ddb9806d33ae/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
-golang.org/x/sys v0.0.0-20200728102440-3e129f6d46b1/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
golang.org/x/sys v0.0.0-20200803210538-64077c9b5642/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
-golang.org/x/sys v0.0.0-20200817155316-9781c653f443/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
golang.org/x/sys v0.0.0-20200905004654-be1d3432aa8f/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
-golang.org/x/sys v0.0.0-20200909081042-eff7692f9009/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
-golang.org/x/sys v0.0.0-20200916030750-2334cc1a136f/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
-golang.org/x/sys v0.0.0-20200922070232-aee5d888a860/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
-golang.org/x/sys v0.0.0-20200923182605-d9f96fdee20d/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
golang.org/x/sys v0.0.0-20200930185726-fdedc70b468f/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
-golang.org/x/sys v0.0.0-20201112073958-5cba982894dd/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
-golang.org/x/sys v0.0.0-20201117170446-d9b008d0a637/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
golang.org/x/sys v0.0.0-20201119102817-f84b799fce68/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
golang.org/x/sys v0.0.0-20201201145000-ef89a241ccb3/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
-golang.org/x/sys v0.0.0-20201202213521-69691e467435/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
golang.org/x/sys v0.0.0-20201204225414-ed752295db88/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
golang.org/x/sys v0.0.0-20210104204734-6f8348627aad/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
golang.org/x/sys v0.0.0-20210119212857-b64e53b001e4/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
@@ -2397,16 +2102,12 @@ golang.org/x/sys v0.0.0-20210220050731-9a76102bfb43/go.mod h1:h1NjWce9XRLGQEsW7w
golang.org/x/sys v0.0.0-20210305230114-8fe3ee5dd75b/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
golang.org/x/sys v0.0.0-20210315160823-c6e025ad8005/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
golang.org/x/sys v0.0.0-20210320140829-1e4c9ba3b0c4/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
-golang.org/x/sys v0.0.0-20210324051608-47abb6519492/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
golang.org/x/sys v0.0.0-20210330210617-4fbd30eecc44/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
golang.org/x/sys v0.0.0-20210403161142-5e06dd20ab57/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
golang.org/x/sys v0.0.0-20210420072515-93ed5bcd2bfe/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
golang.org/x/sys v0.0.0-20210423082822-04245dca01da/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
-golang.org/x/sys v0.0.0-20210423185535-09eb48e85fd7/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
-golang.org/x/sys v0.0.0-20210426230700-d19ff857e887/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
golang.org/x/sys v0.0.0-20210510120138-977fb7262007/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
golang.org/x/sys v0.0.0-20210514084401-e8d321eab015/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
-golang.org/x/sys v0.0.0-20210603081109-ebe580a85c40/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
golang.org/x/sys v0.0.0-20210603125802-9665404d3644/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
golang.org/x/sys v0.0.0-20210615035016-665e8c7367d1/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
golang.org/x/sys v0.0.0-20210616094352-59db8d763f22/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
@@ -2414,13 +2115,10 @@ golang.org/x/sys v0.0.0-20210630005230-0f9fa26af87c/go.mod h1:oPkhp1MJrh7nUepCBc
golang.org/x/sys v0.0.0-20210806184541-e5e7981a1069/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
golang.org/x/sys v0.0.0-20210809222454-d867a43fc93e/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
golang.org/x/sys v0.0.0-20210816183151-1e6c022a8912/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
-golang.org/x/sys v0.0.0-20210819135213-f52c844e1c1c/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
golang.org/x/sys v0.0.0-20210823070655-63515b42dcdf/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
-golang.org/x/sys v0.0.0-20210906170528-6f6e22806c34/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
golang.org/x/sys v0.0.0-20210908233432-aa78b53d3365/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
golang.org/x/sys v0.0.0-20211007075335-d3039528d8ac/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
golang.org/x/sys v0.0.0-20211025201205-69cdffdb9359/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
-golang.org/x/sys v0.0.0-20211116061358-0a5406a5449c/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
golang.org/x/sys v0.0.0-20211124211545-fe61309f8881/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
golang.org/x/sys v0.0.0-20211210111614-af8b64212486/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
golang.org/x/sys v0.0.0-20211216021012-1d35b9e2eb4e/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
@@ -2441,8 +2139,9 @@ golang.org/x/sys v0.0.0-20220722155257-8c9f86f7a55f/go.mod h1:oPkhp1MJrh7nUepCBc
golang.org/x/sys v0.0.0-20220728004956-3c1f35247d10/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
golang.org/x/sys v0.0.0-20220811171246-fbc7d0a398ab/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
golang.org/x/sys v0.0.0-20220825204002-c680a09ffe64/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
+golang.org/x/sys v0.0.0-20220906165534-d0df966e6959/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
golang.org/x/sys v0.0.0-20220908164124-27713097b956/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
-golang.org/x/sys v0.0.0-20220909162455-aba9fc2a8ff2/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
+golang.org/x/sys v0.0.0-20221013171732-95e765b1cc43/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
golang.org/x/sys v0.1.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
golang.org/x/sys v0.2.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
golang.org/x/sys v0.3.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
@@ -2450,18 +2149,20 @@ golang.org/x/sys v0.4.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
golang.org/x/sys v0.5.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
golang.org/x/sys v0.6.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
golang.org/x/sys v0.9.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
-golang.org/x/sys v0.11.0 h1:eG7RXZHdqOJ1i+0lgLgCpSXAp6M3LYlAo6osgSi0xOM=
-golang.org/x/sys v0.11.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
+golang.org/x/sys v0.12.0 h1:CM0HF96J0hcLAwsHPJZjfdNzs0gftsLfgKt57wWHJ0o=
+golang.org/x/sys v0.12.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
golang.org/x/term v0.0.0-20201117132131-f5c789dd3221/go.mod h1:Nr5EML6q2oocZ2LXRh80K7BxOlk5/8JxuGnuhpl+muw=
golang.org/x/term v0.0.0-20201126162022-7de9c90e9dd1/go.mod h1:bj7SfCRtBDWHUb9snDiAeCFNEtKQo2Wmx5Cou7ajbmo=
golang.org/x/term v0.0.0-20210927222741-03fcf44c2211/go.mod h1:jbD1KX2456YbFQfuXm/mYQcufACuNUgVhRMnK/tPxf8=
+golang.org/x/term v0.0.0-20220526004731-065cf7ba2467/go.mod h1:jbD1KX2456YbFQfuXm/mYQcufACuNUgVhRMnK/tPxf8=
golang.org/x/term v0.0.0-20220722155259-a9ba230a4035/go.mod h1:jbD1KX2456YbFQfuXm/mYQcufACuNUgVhRMnK/tPxf8=
golang.org/x/term v0.1.0/go.mod h1:jbD1KX2456YbFQfuXm/mYQcufACuNUgVhRMnK/tPxf8=
golang.org/x/term v0.2.0/go.mod h1:TVmDHMZPmdnySmBfhjOoOdhjzdE1h4u1VwSiw2l1Nuc=
golang.org/x/term v0.4.0/go.mod h1:9P2UbLfCdcvo3p/nzKvsmas4TnlujnuoV9hGgYzW1lQ=
golang.org/x/term v0.5.0/go.mod h1:jMB1sMXY+tzblOD4FWmEbocvup2/aLOaQEp7JmGp78k=
-golang.org/x/term v0.11.0 h1:F9tnn/DA/Im8nCwm+fX+1/eBwi4qFjRT++MhtVC4ZX0=
-golang.org/x/term v0.11.0/go.mod h1:zC9APTIj3jG3FdV/Ons+XE1riIZXG4aZ4GTHiPZJPIU=
+golang.org/x/term v0.6.0/go.mod h1:m6U89DPEgQRMq3DNkDClhWw02AUbt2daBVO4cn4Hv9U=
+golang.org/x/term v0.12.0 h1:/ZfYdc3zq+q02Rv9vGqTeSItdzZTSNDmfTi0mBAuidU=
+golang.org/x/term v0.12.0/go.mod h1:owVbMEjm3cBLCHdkQu9b1opXd4ETQWc3BhuQGKgXgvU=
golang.org/x/text v0.0.0-20170915032832-14c0d48ead0c/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=
@@ -2475,15 +2176,13 @@ golang.org/x/text v0.3.8/go.mod h1:E6s5w1FMmriuDzIBO73fBruAKo1PCIq6d2Q6DHfQ8WQ=
golang.org/x/text v0.4.0/go.mod h1:mrYo+phRRbMaCq/xk9113O4dZlRixOauAjOtrjsXDZ8=
golang.org/x/text v0.6.0/go.mod h1:mrYo+phRRbMaCq/xk9113O4dZlRixOauAjOtrjsXDZ8=
golang.org/x/text v0.7.0/go.mod h1:mrYo+phRRbMaCq/xk9113O4dZlRixOauAjOtrjsXDZ8=
-golang.org/x/text v0.12.0 h1:k+n5B8goJNdU7hSvEtMUz3d1Q6D/XW4COJSJR6fN0mc=
-golang.org/x/text v0.12.0/go.mod h1:TvPlkZtksWOMsz7fbANvkp4WM8x/WCo/om8BMLbz+aE=
-golang.org/x/time v0.0.0-20180412165947-fbb02b2291d2/go.mod h1:tRJNPiyCQ0inRvYxbN9jk5I+vvW/OXSQhTDSoE431IQ=
+golang.org/x/text v0.8.0/go.mod h1:e1OnstbJyHTd6l/uOt8jFFHp6TRDWZR/bV3emEE/zU8=
+golang.org/x/text v0.13.0 h1:ablQoSUd0tRdKxZewP80B+BaqeKJuVhuRxj/dkrun3k=
+golang.org/x/text v0.13.0/go.mod h1:TvPlkZtksWOMsz7fbANvkp4WM8x/WCo/om8BMLbz+aE=
golang.org/x/time v0.0.0-20181108054448-85acf8d2951c/go.mod h1:tRJNPiyCQ0inRvYxbN9jk5I+vvW/OXSQhTDSoE431IQ=
golang.org/x/time v0.0.0-20190308202827-9d24e82272b4/go.mod h1:tRJNPiyCQ0inRvYxbN9jk5I+vvW/OXSQhTDSoE431IQ=
golang.org/x/time v0.0.0-20191024005414-555d28b269f0/go.mod h1:tRJNPiyCQ0inRvYxbN9jk5I+vvW/OXSQhTDSoE431IQ=
golang.org/x/time v0.0.0-20200416051211-89c76fbcd5d1/go.mod h1:tRJNPiyCQ0inRvYxbN9jk5I+vvW/OXSQhTDSoE431IQ=
-golang.org/x/time v0.0.0-20200630173020-3af7569d3a1e/go.mod h1:tRJNPiyCQ0inRvYxbN9jk5I+vvW/OXSQhTDSoE431IQ=
-golang.org/x/time v0.0.0-20210220033141-f8bda1e9f3ba/go.mod h1:tRJNPiyCQ0inRvYxbN9jk5I+vvW/OXSQhTDSoE431IQ=
golang.org/x/time v0.0.0-20220922220347-f3bd1da661af/go.mod h1:tRJNPiyCQ0inRvYxbN9jk5I+vvW/OXSQhTDSoE431IQ=
golang.org/x/time v0.1.0/go.mod h1:tRJNPiyCQ0inRvYxbN9jk5I+vvW/OXSQhTDSoE431IQ=
golang.org/x/time v0.3.0 h1:rg5rLMjNzMS1RkNLzCG38eapWhnYLFYXDXj2gOlr8j4=
@@ -2509,7 +2208,6 @@ golang.org/x/tools v0.0.0-20190614205625-5aca471b1d59/go.mod h1:/rFqwRUd4F7ZHNgw
golang.org/x/tools v0.0.0-20190621195816-6e04913cbbac/go.mod h1:/rFqwRUd4F7ZHNgwSSTFct+R/Kf4OFW1sUzUTQQTgfc=
golang.org/x/tools v0.0.0-20190624222133-a101b041ded4/go.mod h1:/rFqwRUd4F7ZHNgwSSTFct+R/Kf4OFW1sUzUTQQTgfc=
golang.org/x/tools v0.0.0-20190628153133-6cdbf07be9d0/go.mod h1:/rFqwRUd4F7ZHNgwSSTFct+R/Kf4OFW1sUzUTQQTgfc=
-golang.org/x/tools v0.0.0-20190706070813-72ffa07ba3db/go.mod h1:jcCCGcm9btYwXyDqrUWc6MKQKKGJCWEQ3AfLSRIbEuI=
golang.org/x/tools v0.0.0-20190816200558-6889da9d5479/go.mod h1:b+2E5dAYhXwXZwtnZ6UAqBI28+e2cm9otk0dWdXHAEo=
golang.org/x/tools v0.0.0-20190911174233-4f2ddba30aff/go.mod h1:b+2E5dAYhXwXZwtnZ6UAqBI28+e2cm9otk0dWdXHAEo=
golang.org/x/tools v0.0.0-20191012152004-8de300cfc20a/go.mod h1:b+2E5dAYhXwXZwtnZ6UAqBI28+e2cm9otk0dWdXHAEo=
@@ -2534,6 +2232,7 @@ golang.org/x/tools v0.0.0-20200224181240-023911ca70b2/go.mod h1:TB2adYChydJhpapK
golang.org/x/tools v0.0.0-20200227222343-706bc42d1f0d/go.mod h1:TB2adYChydJhpapKDTa4BR/hXlZSLoq2Wpct/0txZ28=
golang.org/x/tools v0.0.0-20200304193943-95d2e580d8eb/go.mod h1:o4KQGtdN14AW+yjsvvwRTJJuXz8XRtIHtEnmAXLyFUw=
golang.org/x/tools v0.0.0-20200312045724-11d5b4c81c7d/go.mod h1:o4KQGtdN14AW+yjsvvwRTJJuXz8XRtIHtEnmAXLyFUw=
+golang.org/x/tools v0.0.0-20200313205530-4303120df7d8/go.mod h1:Sl4aGygMT6LrqrWclx+PTx3U+LnKx/seiNR+3G19Ar8=
golang.org/x/tools v0.0.0-20200331025713-a30bf2db82d4/go.mod h1:Sl4aGygMT6LrqrWclx+PTx3U+LnKx/seiNR+3G19Ar8=
golang.org/x/tools v0.0.0-20200501065659-ab2804fb9c9d/go.mod h1:EkVYQZoAsY45+roYkvgYkIh4xh/qjgUK9TdY2XT94GE=
golang.org/x/tools v0.0.0-20200505023115-26f46d2f7ef8/go.mod h1:EkVYQZoAsY45+roYkvgYkIh4xh/qjgUK9TdY2XT94GE=
@@ -2546,7 +2245,6 @@ golang.org/x/tools v0.0.0-20200729194436-6467de6f59a7/go.mod h1:njjCfa9FT2d7l9Bc
golang.org/x/tools v0.0.0-20200804011535-6c149bb5ef0d/go.mod h1:njjCfa9FT2d7l9Bc6FUM5FLjQPp3cFF28FI3qnDFljA=
golang.org/x/tools v0.0.0-20200825202427-b303f430e36d/go.mod h1:njjCfa9FT2d7l9Bc6FUM5FLjQPp3cFF28FI3qnDFljA=
golang.org/x/tools v0.0.0-20200904185747-39188db58858/go.mod h1:Cj7w3i3Rnn0Xh82ur9kSqwfTHTeVxaDqrfMjpcNT6bE=
-golang.org/x/tools v0.0.0-20200916195026-c9a70fc28ce3/go.mod h1:z6u4i615ZeAfBE4XtMziQW1fSVJXACjjbWkB/mvPzlU=
golang.org/x/tools v0.0.0-20201110124207-079ba7bd75cd/go.mod h1:emZCQorbCU4vsT4fOWvOPXz4eW1wZW4PmDk9uLelYpA=
golang.org/x/tools v0.0.0-20201201161351-ac6f37ff4c2a/go.mod h1:emZCQorbCU4vsT4fOWvOPXz4eW1wZW4PmDk9uLelYpA=
golang.org/x/tools v0.0.0-20201208233053-a543418bbed2/go.mod h1:emZCQorbCU4vsT4fOWvOPXz4eW1wZW4PmDk9uLelYpA=
@@ -2562,6 +2260,7 @@ golang.org/x/tools v0.1.7/go.mod h1:LGqMHiF4EqQNHR1JncWGqT5BVaXmza+X+BDGol+dOxo=
golang.org/x/tools v0.1.12/go.mod h1:hNGJHUnrk76NpqgfD5Aqm5Crs+Hm0VOH/i9J2+nxYbc=
golang.org/x/tools v0.2.0/go.mod h1:y4OqIKeOV/fWJetJ8bXPU1sEVniLMIyDAZWeHdV+NTA=
golang.org/x/tools v0.3.0/go.mod h1:/rWhSS2+zyEVwoJf8YAX6L2f0ntZ7Kn/mGgAWcipA5k=
+golang.org/x/tools v0.6.0/go.mod h1:Xwgl3UAJ/d3gWutnCtw505GrjyAbvKui8lOU390QaIU=
golang.org/x/tools v0.9.3 h1:Gn1I8+64MsuTb/HpH+LmQtNas23LhUVr3rYZ0eKuaMM=
golang.org/x/tools v0.9.3/go.mod h1:owI94Op576fPu3cIGQeHs3joujW/2Oc6MtlxbF5dfNc=
golang.org/x/xerrors v0.0.0-20190717185122-a985d3407aa7/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0=
@@ -2573,8 +2272,8 @@ golang.org/x/xerrors v0.0.0-20220517211312-f3a8303e98df/go.mod h1:K8+ghG5WaK9qNq
golang.org/x/xerrors v0.0.0-20220609144429-65e65417b02f/go.mod h1:K8+ghG5WaK9qNqU5K3HdILfMLy1f3aNYFI/wnl100a8=
golang.org/x/xerrors v0.0.0-20220907171357-04be3eba64a2 h1:H2TDz8ibqkAF6YGhCdN3jS9O0/s90v0rJh3X/OLHEUk=
golang.org/x/xerrors v0.0.0-20220907171357-04be3eba64a2/go.mod h1:K8+ghG5WaK9qNqU5K3HdILfMLy1f3aNYFI/wnl100a8=
-gomodules.xyz/jsonpatch/v2 v2.2.0 h1:4pT439QV83L+G9FkcCriY6EkpcK6r6bK+A5FBUMI7qY=
-gomodules.xyz/jsonpatch/v2 v2.2.0/go.mod h1:WXp+iVDkoLQqPudfQ9GBlwB2eZ5DKOnjQZCYdOS8GPY=
+gomodules.xyz/jsonpatch/v2 v2.3.0 h1:8NFhfS6gzxNqjLIYnZxg319wZ5Qjnx4m/CcX+Klzazc=
+gomodules.xyz/jsonpatch/v2 v2.3.0/go.mod h1:AH3dM2RI6uoBZxn3LVrfvJ3E0/9dG4cSrbuBJT4moAY=
google.golang.org/api v0.4.0/go.mod h1:8k5glujaEP+g9n7WNsDg8QP6cUVNI86fCNMcbazEtwE=
google.golang.org/api v0.7.0/go.mod h1:WtwebWUNSVBH/HAw79HIFXZNqEvBhG+Ra+ax0hx3E3M=
google.golang.org/api v0.8.0/go.mod h1:o4eAsZoiT+ibD93RtjEohWalFOjRDx6CVaqeizhEnKg=
@@ -2627,8 +2326,8 @@ google.golang.org/api v0.99.0/go.mod h1:1YOf74vkVndF7pG6hIHuINsM7eWwpVTAfNMNiL91
google.golang.org/api v0.100.0/go.mod h1:ZE3Z2+ZOr87Rx7dqFsdRQkRBk36kDtp/h+QpHbB7a70=
google.golang.org/api v0.102.0/go.mod h1:3VFl6/fzoA+qNuS1N1/VfXY4LjoXN/wzeIp7TweWwGo=
google.golang.org/api v0.103.0/go.mod h1:hGtW6nK1AC+d9si/UBhw8Xli+QMOf6xyNAyJw4qU9w0=
-google.golang.org/api v0.122.0 h1:zDobeejm3E7pEG1mNHvdxvjs5XJoCMzyNH+CmwL94Es=
-google.golang.org/api v0.122.0/go.mod h1:gcitW0lvnyWjSp9nKxAbdHKIZ6vF4aajGueeslZOyms=
+google.golang.org/api v0.124.0 h1:dP6Ef1VgOGqQ8eiv4GiY8RhmeyqzovcXBYPDUYG8Syo=
+google.golang.org/api v0.124.0/go.mod h1:xu2HQurE5gi/3t1aFCvhPD781p0a3p11sdunTJ2BlP4=
google.golang.org/appengine v1.1.0/go.mod h1:EbEs0AVv82hx2wNQdGPgUI5lhzA/G0D9YwlJXL52JkM=
google.golang.org/appengine v1.4.0/go.mod h1:xpcJRLb0r/rnEns0DIKYYv+WjYCduHsrkT7/EB5XEv4=
google.golang.org/appengine v1.5.0/go.mod h1:xpcJRLb0r/rnEns0DIKYYv+WjYCduHsrkT7/EB5XEv4=
@@ -2642,7 +2341,6 @@ google.golang.org/genproto v0.0.0-20190307195333-5fe7a883aa19/go.mod h1:VzzqZJRn
google.golang.org/genproto v0.0.0-20190418145605-e7d98fc518a7/go.mod h1:VzzqZJRnGkLBvHegQrXjBqPurQTc5/KpmUdxsrq26oE=
google.golang.org/genproto v0.0.0-20190425155659-357c62f0e4bb/go.mod h1:VzzqZJRnGkLBvHegQrXjBqPurQTc5/KpmUdxsrq26oE=
google.golang.org/genproto v0.0.0-20190502173448-54afdca5d873/go.mod h1:VzzqZJRnGkLBvHegQrXjBqPurQTc5/KpmUdxsrq26oE=
-google.golang.org/genproto v0.0.0-20190522204451-c2c4e71fbf69/go.mod h1:z3L6/3dTEVtUr6QSP8miRzeRqwQOioJ9I66odjN4I7s=
google.golang.org/genproto v0.0.0-20190801165951-fa694d86fc64/go.mod h1:DMBHOl98Agz4BDEuKkezgsaosCRResVns1a3J2ZsMNc=
google.golang.org/genproto v0.0.0-20190819201941-24fa4b261c55/go.mod h1:DMBHOl98Agz4BDEuKkezgsaosCRResVns1a3J2ZsMNc=
google.golang.org/genproto v0.0.0-20190911173649-1774047e7e51/go.mod h1:IbNlFCBrqXvoKpeg0TB2l7cyZUmoaFKYIwrEpbDKLA8=
@@ -2651,7 +2349,6 @@ google.golang.org/genproto v0.0.0-20191115194625-c23dd37a84c9/go.mod h1:n3cpQtvx
google.golang.org/genproto v0.0.0-20191216164720-4f79533eabd1/go.mod h1:n3cpQtvxv34hfy77yVDNjmbRyujviMdxYliBSkLhpCc=
google.golang.org/genproto v0.0.0-20191230161307-f3c370f40bfb/go.mod h1:n3cpQtvxv34hfy77yVDNjmbRyujviMdxYliBSkLhpCc=
google.golang.org/genproto v0.0.0-20200115191322-ca5a22157cba/go.mod h1:n3cpQtvxv34hfy77yVDNjmbRyujviMdxYliBSkLhpCc=
-google.golang.org/genproto v0.0.0-20200117163144-32f20d992d24/go.mod h1:n3cpQtvxv34hfy77yVDNjmbRyujviMdxYliBSkLhpCc=
google.golang.org/genproto v0.0.0-20200122232147-0452cf42e150/go.mod h1:n3cpQtvxv34hfy77yVDNjmbRyujviMdxYliBSkLhpCc=
google.golang.org/genproto v0.0.0-20200204135345-fa8e72b47b90/go.mod h1:GmwEX6Z4W5gMy59cAlVYjN9JhxgbQH6Gn+gFDQe2lzA=
google.golang.org/genproto v0.0.0-20200212174721-66ed5ce911ce/go.mod h1:55QSHmfGQM9UVYDPBsyGGes0y52j32PQ3BqQfXhyH3c=
@@ -2666,14 +2363,12 @@ google.golang.org/genproto v0.0.0-20200511104702-f5ebc3bea380/go.mod h1:55QSHmfG
google.golang.org/genproto v0.0.0-20200513103714-09dca8ec2884/go.mod h1:55QSHmfGQM9UVYDPBsyGGes0y52j32PQ3BqQfXhyH3c=
google.golang.org/genproto v0.0.0-20200515170657-fc4c6c6a6587/go.mod h1:YsZOwe1myG/8QRHRsmBRE1LrgQY60beZKjly0O1fX9U=
google.golang.org/genproto v0.0.0-20200526211855-cb27e3aa2013/go.mod h1:NbSheEEYHJ7i3ixzK3sjbqSGDJWnxyFXZblF3eUsNvo=
-google.golang.org/genproto v0.0.0-20200527145253-8367513e4ece/go.mod h1:jDfRM7FcilCzHH/e9qn6dsT145K34l5v+OpcnNgKAAA=
google.golang.org/genproto v0.0.0-20200618031413-b414f8b61790/go.mod h1:jDfRM7FcilCzHH/e9qn6dsT145K34l5v+OpcnNgKAAA=
google.golang.org/genproto v0.0.0-20200729003335-053ba62fc06f/go.mod h1:FWY/as6DDZQgahTzZj3fqbO1CbirC29ZNUFHwi0/+no=
google.golang.org/genproto v0.0.0-20200804131852-c06518451d9c/go.mod h1:FWY/as6DDZQgahTzZj3fqbO1CbirC29ZNUFHwi0/+no=
google.golang.org/genproto v0.0.0-20200825200019-8632dd797987/go.mod h1:FWY/as6DDZQgahTzZj3fqbO1CbirC29ZNUFHwi0/+no=
google.golang.org/genproto v0.0.0-20200904004341-0bd0a958aa1d/go.mod h1:FWY/as6DDZQgahTzZj3fqbO1CbirC29ZNUFHwi0/+no=
google.golang.org/genproto v0.0.0-20201109203340-2640f1f9cdfb/go.mod h1:FWY/as6DDZQgahTzZj3fqbO1CbirC29ZNUFHwi0/+no=
-google.golang.org/genproto v0.0.0-20201110150050-8816d57aaa9a/go.mod h1:FWY/as6DDZQgahTzZj3fqbO1CbirC29ZNUFHwi0/+no=
google.golang.org/genproto v0.0.0-20201201144952-b05cb90ed32e/go.mod h1:FWY/as6DDZQgahTzZj3fqbO1CbirC29ZNUFHwi0/+no=
google.golang.org/genproto v0.0.0-20201210142538-e3217bee35cc/go.mod h1:FWY/as6DDZQgahTzZj3fqbO1CbirC29ZNUFHwi0/+no=
google.golang.org/genproto v0.0.0-20201214200347-8c77b98c765d/go.mod h1:FWY/as6DDZQgahTzZj3fqbO1CbirC29ZNUFHwi0/+no=
@@ -2703,7 +2398,6 @@ google.golang.org/genproto v0.0.0-20211118181313-81c1377c94b1/go.mod h1:5CzLGKJ6
google.golang.org/genproto v0.0.0-20211206160659-862468c7d6e0/go.mod h1:5CzLGKJ67TSI2B9POpiiyGha0AjJvZIUgRMt1dSmuhc=
google.golang.org/genproto v0.0.0-20211208223120-3a66f561d7aa/go.mod h1:5CzLGKJ67TSI2B9POpiiyGha0AjJvZIUgRMt1dSmuhc=
google.golang.org/genproto v0.0.0-20211221195035-429b39de9b1c/go.mod h1:5CzLGKJ67TSI2B9POpiiyGha0AjJvZIUgRMt1dSmuhc=
-google.golang.org/genproto v0.0.0-20220107163113-42d7afdf6368/go.mod h1:5CzLGKJ67TSI2B9POpiiyGha0AjJvZIUgRMt1dSmuhc=
google.golang.org/genproto v0.0.0-20220126215142-9970aeb2e350/go.mod h1:5CzLGKJ67TSI2B9POpiiyGha0AjJvZIUgRMt1dSmuhc=
google.golang.org/genproto v0.0.0-20220207164111-0872dc986b00/go.mod h1:5CzLGKJ67TSI2B9POpiiyGha0AjJvZIUgRMt1dSmuhc=
google.golang.org/genproto v0.0.0-20220218161850-94dd64e39d7c/go.mod h1:kGP+zUP2Ddo0ayMi4YuN7C3WZyJvGLZRh8Z5wnAqvEI=
@@ -2754,8 +2448,12 @@ google.golang.org/genproto v0.0.0-20221118155620-16455021b5e6/go.mod h1:rZS5c/ZV
google.golang.org/genproto v0.0.0-20221201164419-0e50fba7f41c/go.mod h1:rZS5c/ZVYMaOGBfO68GWtjOw/eLaZM1X6iVtgjZ+EWg=
google.golang.org/genproto v0.0.0-20221202195650-67e5cbc046fd/go.mod h1:cTsE614GARnxrLsqKREzmNYJACSWWpAWdNMwnD7c2BE=
google.golang.org/genproto v0.0.0-20230110181048-76db0878b65f/go.mod h1:RGgjbofJ8xD9Sq1VVhDM1Vok1vRONV+rg+CjzG4SZKM=
-google.golang.org/genproto v0.0.0-20230410155749-daa745c078e1 h1:KpwkzHKEF7B9Zxg18WzOa7djJ+Ha5DzthMyZYQfEn2A=
-google.golang.org/genproto v0.0.0-20230410155749-daa745c078e1/go.mod h1:nKE/iIaLqn2bQwXBg8f1g2Ylh6r5MN5CmZvuzZCgsCU=
+google.golang.org/genproto v0.0.0-20230526161137-0005af68ea54 h1:9NWlQfY2ePejTmfwUH1OWwmznFa+0kKcHGPDvcPza9M=
+google.golang.org/genproto v0.0.0-20230526161137-0005af68ea54/go.mod h1:zqTuNwFlFRsw5zIts5VnzLQxSRqh+CGOTVMlYbY0Eyk=
+google.golang.org/genproto/googleapis/api v0.0.0-20230525234035-dd9d682886f9 h1:m8v1xLLLzMe1m5P+gCTF8nJB9epwZQUBERm20Oy1poQ=
+google.golang.org/genproto/googleapis/api v0.0.0-20230525234035-dd9d682886f9/go.mod h1:vHYtlOoi6TsQ3Uk2yxR7NI5z8uoV+3pZtR4jmHIkRig=
+google.golang.org/genproto/googleapis/rpc v0.0.0-20230525234030-28d5490b6b19 h1:0nDDozoAU19Qb2HwhXadU8OcsiO/09cnTqhUtq2MEOM=
+google.golang.org/genproto/googleapis/rpc v0.0.0-20230525234030-28d5490b6b19/go.mod h1:66JfowdXAEgad5O9NnYcsNPLCPZJD++2L9X0PCMODrA=
google.golang.org/grpc v1.53.0 h1:LAv2ds7cmFV/XTS3XG1NneeENYrXGmorPxsBbptIjNc=
google.golang.org/grpc v1.53.0/go.mod h1:OnIrk0ipVdj4N5d9IUoFUx72/VlD7+jUsHwZgwSMQpw=
google.golang.org/grpc/cmd/protoc-gen-go-grpc v1.1.0/go.mod h1:6Kw0yEErY5E/yWrBtf03jp27GLLJujG4z/JK95pnjjw=
@@ -2789,26 +2487,23 @@ gopkg.in/check.v1 v1.0.0-20200227125254-8fa46927fb4f/go.mod h1:Co6ibVJAznAaIkqp8
gopkg.in/check.v1 v1.0.0-20200902074654-038fdea0a05b/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0=
gopkg.in/check.v1 v1.0.0-20201130134442-10cb98267c6c h1:Hei/4ADfdWqJk1ZMxUNpqntNwaWcugrBjAiHlqqRiVk=
gopkg.in/check.v1 v1.0.0-20201130134442-10cb98267c6c/go.mod h1:JHkPIbrfpd72SG/EVd6muEfDQjcINNoR0C8j2r3qZ4Q=
-gopkg.in/cheggaaa/pb.v1 v1.0.25/go.mod h1:V/YB90LKu/1FcN3WVnfiiE5oMCibMjukxqG/qStrOgw=
gopkg.in/cheggaaa/pb.v1 v1.0.27/go.mod h1:V/YB90LKu/1FcN3WVnfiiE5oMCibMjukxqG/qStrOgw=
-gopkg.in/cheggaaa/pb.v1 v1.0.28 h1:n1tBJnnK2r7g9OW2btFH91V92STTUevLXYFb8gy9EMk=
-gopkg.in/cheggaaa/pb.v1 v1.0.28/go.mod h1:V/YB90LKu/1FcN3WVnfiiE5oMCibMjukxqG/qStrOgw=
gopkg.in/errgo.v2 v2.1.0/go.mod h1:hNsd1EY+bozCKY1Ytp96fpM3vjJbqLJn88ws8XvfDNI=
gopkg.in/fsnotify.v1 v1.4.7/go.mod h1:Tz8NjZHkW78fSQdbUxIjBTcgA1z1m8ZHf0WmKUhAMys=
gopkg.in/gemnasium/logrus-airbrake-hook.v2 v2.1.2/go.mod h1:Xk6kEKp8OKb+X14hQBKWaSkCsqBpgog8nAV2xsGOxlo=
+gopkg.in/go-jose/go-jose.v2 v2.6.1 h1:qEzJlIDmG9q5VO0M/o8tGS65QMHMS1w01TQJB1VPJ4U=
+gopkg.in/go-jose/go-jose.v2 v2.6.1/go.mod h1:zzZDPkNNw/c9IE7Z9jr11mBZQhKQTMzoEEIoEdZlFBI=
gopkg.in/inf.v0 v0.9.1 h1:73M5CoZyi3ZLMOyDlQh031Cx6N9NDJ2Vvfl76EDAgDc=
gopkg.in/inf.v0 v0.9.1/go.mod h1:cWUDdTG/fYaXco+Dcufb5Vnc6Gp2YChqWtbxRZE0mXw=
gopkg.in/ini.v1 v1.51.0/go.mod h1:pNLf8WUiyNEtQjuu5G5vTm06TEv9tsIgeAvK8hOrP4k=
gopkg.in/ini.v1 v1.62.0/go.mod h1:pNLf8WUiyNEtQjuu5G5vTm06TEv9tsIgeAvK8hOrP4k=
gopkg.in/ini.v1 v1.67.0 h1:Dgnx+6+nfE+IfzjUEISNeydPJh9AXNNsWbGP9KzCsOA=
gopkg.in/ini.v1 v1.67.0/go.mod h1:pNLf8WUiyNEtQjuu5G5vTm06TEv9tsIgeAvK8hOrP4k=
-gopkg.in/natefinch/lumberjack.v2 v2.0.0 h1:1Lc07Kr7qY4U2YPouBjpCLxpiyxIVoxqXgkXLknAOE8=
-gopkg.in/natefinch/lumberjack.v2 v2.0.0/go.mod h1:l0ndWWf7gzL7RNwBG7wST/UCcT4T24xpD6X8LsfU/+k=
+gopkg.in/natefinch/lumberjack.v2 v2.2.1 h1:bBRl1b0OH9s/DuPhuXpNl+VtCaJXFZ5/uEFST95x9zc=
+gopkg.in/natefinch/lumberjack.v2 v2.2.1/go.mod h1:YD8tP3GAjkrDg1eZH7EGmyESg/lsYskCTPBJVb9jqSc=
gopkg.in/resty.v1 v1.12.0/go.mod h1:mDo4pnntr5jdWRML875a/NmxYqAlA73dVijT2AXvQQo=
gopkg.in/rethinkdb/rethinkdb-go.v6 v6.2.1 h1:d4KQkxAaAiRY2h5Zqis161Pv91A37uZyJOx73duwUwM=
gopkg.in/rethinkdb/rethinkdb-go.v6 v6.2.1/go.mod h1:WbjuEoo1oadwzQ4apSDU+JTvmllEHtsNHS6y7vFc7iw=
-gopkg.in/square/go-jose.v2 v2.2.2/go.mod h1:M9dMgbHiYLoDGQrXy7OpJDJWiKiU//h+vD76mk0e1AI=
-gopkg.in/square/go-jose.v2 v2.3.1/go.mod h1:M9dMgbHiYLoDGQrXy7OpJDJWiKiU//h+vD76mk0e1AI=
gopkg.in/square/go-jose.v2 v2.5.1/go.mod h1:M9dMgbHiYLoDGQrXy7OpJDJWiKiU//h+vD76mk0e1AI=
gopkg.in/square/go-jose.v2 v2.6.0 h1:NGk74WTnPKBNUhNzQX7PYcTLUjoq7mzKk2OKbvwk2iI=
gopkg.in/square/go-jose.v2 v2.6.0/go.mod h1:M9dMgbHiYLoDGQrXy7OpJDJWiKiU//h+vD76mk0e1AI=
@@ -2821,7 +2516,6 @@ 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.3/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.5/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI=
gopkg.in/yaml.v2 v2.2.8/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI=
gopkg.in/yaml.v2 v2.3.0/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI=
gopkg.in/yaml.v2 v2.4.0 h1:D8xgwECY7CYvx+Y2n4sBz93Jn9JRvxdiyyo8CTfuKaY=
@@ -2835,11 +2529,10 @@ gopkg.in/yaml.v3 v3.0.1 h1:fxVm/GzAzEWqLHuvctI91KS9hhNmmWOoWu0XTYJS7CA=
gopkg.in/yaml.v3 v3.0.1/go.mod h1:K4uyk7z7BCEPqu6E+C64Yfv1cQ7kz7rIZviUmN+EgEM=
gotest.tools v2.2.0+incompatible h1:VsBPFP1AI068pPrMxtb/S8Zkgf9xEmTLJjfM+P5UIEo=
gotest.tools v2.2.0+incompatible/go.mod h1:DsYFclhRJ6vuDpmuTbkuFWG+y2sxOXAzmJt81HFBacw=
-gotest.tools/v3 v3.0.2/go.mod h1:3SzNCllyD9/Y+b5r9JIKQ474KzkZyqLqEfYqMsX94Bk=
-gotest.tools/v3 v3.0.3 h1:4AuOwCGf4lLR9u3YOe2awrHygurzhO/HeQ6laiA6Sx0=
gotest.tools/v3 v3.0.3/go.mod h1:Z7Lb0S5l+klDB31fvDQX8ss/FlKDxtlFlw3Oa8Ymbl8=
-helm.sh/helm/v3 v3.11.1 h1:cmL9fFohOoNQf+wnp2Wa0OhNFH0KFnSzEkVxi3fcc3I=
-helm.sh/helm/v3 v3.11.1/go.mod h1:z/Bu/BylToGno/6dtNGuSmjRqxKq5gaH+FU0BPO+AQ8=
+gotest.tools/v3 v3.4.0 h1:ZazjZUfuVeZGLAmlKKuyv3IKP5orXcwtOwDQH6YVr6o=
+helm.sh/helm/v3 v3.12.3 h1:5y1+Sbty12t48T/t/CGNYUIME5BJ0WKfmW/sobYqkFg=
+helm.sh/helm/v3 v3.12.3/go.mod h1:KPKQiX9IP5HX7o5YnnhViMnNuKiL/lJBVQ47GHe1R0k=
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-20190418001031-e561f6794a2a/go.mod h1:rf3lG4BRIbNafJWhAfAdb/ePZxsR/4RtNHQocxwk9r4=
@@ -2847,82 +2540,55 @@ honnef.co/go/tools v0.0.0-20190523083050-ea95bdfd59fc/go.mod h1:rf3lG4BRIbNafJWh
honnef.co/go/tools v0.0.1-2019.2.3/go.mod h1:a3bituU0lyd329TUQxRnasdCoJDkEUEAqEt0JzvZhAg=
honnef.co/go/tools v0.0.1-2020.1.3/go.mod h1:X/FiERA/W4tHapMX5mGpAtMSVEeEUOyHaw9vFzvIQ3k=
honnef.co/go/tools v0.0.1-2020.1.4/go.mod h1:X/FiERA/W4tHapMX5mGpAtMSVEeEUOyHaw9vFzvIQ3k=
-inet.af/netaddr v0.0.0-20220617031823-097006376321 h1:B4dC8ySKTQXasnjDTMsoCMf1sQG4WsMej0WXaHxunmU=
-inet.af/netaddr v0.0.0-20220617031823-097006376321/go.mod h1:OIezDfdzOgFhuw4HuWapWq2e9l0H9tK4F1j+ETRtF3k=
k8s.io/api v0.19.0/go.mod h1:I1K45XlvTrDjmj5LoM5LuP/KYrhWbjUKT/SoPG0qTjw=
-k8s.io/api v0.20.1/go.mod h1:KqwcCVogGxQY3nBlRpwt+wpAMF/KjaCc7RpywacvqUo=
-k8s.io/api v0.20.4/go.mod h1:++lNL1AJMkDymriNniQsWRkMDzRaX2Y/POTUi8yvqYQ=
-k8s.io/api v0.20.6/go.mod h1:X9e8Qag6JV/bL5G6bU8sdVRltWKmdHsFUGS3eVndqE8=
-k8s.io/api v0.26.3 h1:emf74GIQMTik01Aum9dPP0gAypL8JTLl/lHa4V9RFSU=
-k8s.io/api v0.26.3/go.mod h1:PXsqwPMXBSBcL1lJ9CYDKy7kIReUydukS5JiRlxC3qE=
-k8s.io/apiextensions-apiserver v0.26.3 h1:5PGMm3oEzdB1W/FTMgGIDmm100vn7IaUP5er36dB+YE=
-k8s.io/apiextensions-apiserver v0.26.3/go.mod h1:jdA5MdjNWGP+njw1EKMZc64xAT5fIhN6VJrElV3sfpQ=
+k8s.io/api v0.28.2 h1:9mpl5mOb6vXZvqbQmankOfPIGiudghwCoLl1EYfUZbw=
+k8s.io/api v0.28.2/go.mod h1:RVnJBsjU8tcMq7C3iaRSGMeaKt2TWEUXcpIt/90fjEg=
+k8s.io/apiextensions-apiserver v0.28.1 h1:l2ThkBRjrWpw4f24uq0Da2HaEgqJZ7pcgiEUTKSmQZw=
+k8s.io/apiextensions-apiserver v0.28.1/go.mod h1:sVvrI+P4vxh2YBBcm8n2ThjNyzU4BQGilCQ/JAY5kGs=
k8s.io/apimachinery v0.19.0/go.mod h1:DnPGDnARWFvYa3pMHgSxtbZb7gpzzAZ1pTfaUNDVlmA=
-k8s.io/apimachinery v0.20.1/go.mod h1:WlLqWAHZGg07AeltaI0MV5uk1Omp8xaN0JGLY6gkRpU=
-k8s.io/apimachinery v0.20.4/go.mod h1:WlLqWAHZGg07AeltaI0MV5uk1Omp8xaN0JGLY6gkRpU=
-k8s.io/apimachinery v0.20.6/go.mod h1:ejZXtW1Ra6V1O5H8xPBGz+T3+4gfkTCeExAHKU57MAc=
-k8s.io/apimachinery v0.26.3 h1:dQx6PNETJ7nODU3XPtrwkfuubs6w7sX0M8n61zHIV/k=
-k8s.io/apimachinery v0.26.3/go.mod h1:ats7nN1LExKHvJ9TmwootT00Yz05MuYqPXEXaVeOy5I=
-k8s.io/apiserver v0.20.1/go.mod h1:ro5QHeQkgMS7ZGpvf4tSMx6bBOgPfE+f52KwvXfScaU=
-k8s.io/apiserver v0.20.4/go.mod h1:Mc80thBKOyy7tbvFtB4kJv1kbdD0eIH8k8vianJcbFM=
-k8s.io/apiserver v0.20.6/go.mod h1:QIJXNt6i6JB+0YQRNcS0hdRHJlMhflFmsBDeSgT1r8Q=
-k8s.io/apiserver v0.26.3 h1:blBpv+yOiozkPH2aqClhJmJY+rp53Tgfac4SKPDJnU4=
-k8s.io/apiserver v0.26.3/go.mod h1:CJe/VoQNcXdhm67EvaVjYXxR3QyfwpceKPuPaeLibTA=
-k8s.io/cli-runtime v0.26.3 h1:3ULe0oI28xmgeLMVXIstB+ZL5CTGvWSMVMLeHxitIuc=
-k8s.io/cli-runtime v0.26.3/go.mod h1:5YEhXLV4kLt/OSy9yQwtSSNZU2Z7aTEYta1A+Jg4VC4=
+k8s.io/apimachinery v0.28.2 h1:KCOJLrc6gu+wV1BYgwik4AF4vXOlVJPdiqn0yAWWwXQ=
+k8s.io/apimachinery v0.28.2/go.mod h1:RdzF87y/ngqk9H4z3EL2Rppv5jj95vGS/HaFXrLDApU=
+k8s.io/apiserver v0.28.1 h1:dw2/NKauDZCnOUAzIo2hFhtBRUo6gQK832NV8kuDbGM=
+k8s.io/apiserver v0.28.1/go.mod h1:d8aizlSRB6yRgJ6PKfDkdwCy2DXt/d1FDR6iJN9kY1w=
+k8s.io/cli-runtime v0.28.2 h1:64meB2fDj10/ThIMEJLO29a1oujSm0GQmKzh1RtA/uk=
+k8s.io/cli-runtime v0.28.2/go.mod h1:bTpGOvpdsPtDKoyfG4EG041WIyFZLV9qq4rPlkyYfDA=
k8s.io/client-go v0.19.0/go.mod h1:H9E/VT95blcFQnlyShFgnFT9ZnJOAceiUHM3MlRC+mU=
-k8s.io/client-go v0.20.1/go.mod h1:/zcHdt1TeWSd5HoUe6elJmHSQ6uLLgp4bIJHVEuy+/Y=
-k8s.io/client-go v0.20.4/go.mod h1:LiMv25ND1gLUdBeYxBIwKpkSC5IsozMMmOOeSJboP+k=
-k8s.io/client-go v0.20.6/go.mod h1:nNQMnOvEUEsOzRRFIIkdmYOjAZrC8bgq0ExboWSU1I0=
-k8s.io/client-go v0.26.3 h1:k1UY+KXfkxV2ScEL3gilKcF7761xkYsSD6BC9szIu8s=
-k8s.io/client-go v0.26.3/go.mod h1:ZPNu9lm8/dbRIPAgteN30RSXea6vrCpFvq+MateTUuQ=
+k8s.io/client-go v0.28.2 h1:DNoYI1vGq0slMBN/SWKMZMw0Rq+0EQW6/AK4v9+3VeY=
+k8s.io/client-go v0.28.2/go.mod h1:sMkApowspLuc7omj1FOSUxSoqjr+d5Q0Yc0LOFnYFJY=
k8s.io/code-generator v0.19.0/go.mod h1:moqLn7w0t9cMs4+5CQyxnfA/HV8MF6aAVENF+WZZhgk=
-k8s.io/code-generator v0.19.7/go.mod h1:lwEq3YnLYb/7uVXLorOJfxg+cUu2oihFhHZ0n9NIla0=
-k8s.io/code-generator v0.26.3 h1:DNYPsWoeFwmg4qFg97Z1cHSSv7KSG10mAEIFoZGTQM8=
-k8s.io/code-generator v0.26.3/go.mod h1:ryaiIKwfxEJEaywEzx3dhWOydpVctKYbqLajJf0O8dI=
-k8s.io/component-base v0.20.1/go.mod h1:guxkoJnNoh8LNrbtiQOlyp2Y2XFCZQmrcg2n/DeYNLk=
-k8s.io/component-base v0.20.4/go.mod h1:t4p9EdiagbVCJKrQ1RsA5/V4rFQNDfRlevJajlGwgjI=
-k8s.io/component-base v0.20.6/go.mod h1:6f1MPBAeI+mvuts3sIdtpjljHWBQ2cIy38oBIWMYnrM=
-k8s.io/component-base v0.26.3 h1:oC0WMK/ggcbGDTkdcqefI4wIZRYdK3JySx9/HADpV0g=
-k8s.io/component-base v0.26.3/go.mod h1:5kj1kZYwSC6ZstHJN7oHBqcJC6yyn41eR+Sqa/mQc8E=
-k8s.io/component-helpers v0.26.0 h1:KNgwqs3EUdK0HLfW4GhnbD+q/Zl9U021VfIU7qoVYFk=
-k8s.io/component-helpers v0.26.0/go.mod h1:jHN01qS/Jdj95WCbTe9S2VZ9yxpxXNY488WjF+yW4fo=
-k8s.io/cri-api v0.17.3/go.mod h1:X1sbHmuXhwaHs9xxYffLqJogVsnI+f6cPRcgPel7ywM=
-k8s.io/cri-api v0.20.1/go.mod h1:2JRbKt+BFLTjtrILYVqQK5jqhI+XNdF6UiGMgczeBCI=
-k8s.io/cri-api v0.20.4/go.mod h1:2JRbKt+BFLTjtrILYVqQK5jqhI+XNdF6UiGMgczeBCI=
-k8s.io/cri-api v0.20.6/go.mod h1:ew44AjNXwyn1s0U4xCKGodU7J1HzBeZ1MpGrpa5r8Yc=
+k8s.io/code-generator v0.28.2 h1:u47guga1rCWLnEnffF09p+cqj8B20oHOLoQ1lb1HGtQ=
+k8s.io/code-generator v0.28.2/go.mod h1:ueeSJZJ61NHBa0ccWLey6mwawum25vX61nRZ6WOzN9A=
+k8s.io/component-base v0.28.2 h1:Yc1yU+6AQSlpJZyvehm/NkJBII72rzlEsd6MkBQ+G0E=
+k8s.io/component-base v0.28.2/go.mod h1:4IuQPQviQCg3du4si8GpMrhAIegxpsgPngPRR/zWpzc=
+k8s.io/component-helpers v0.28.2 h1:r/XJ265PMirW9EcGXr/F+2yWrLPo2I69KdvcY/h9HAo=
+k8s.io/component-helpers v0.28.2/go.mod h1:pF1R5YWQ+sgf0i6EbVm+MQCzkYuqutDUibdrkvAa6aI=
k8s.io/cri-api v0.27.1 h1:KWO+U8MfI9drXB/P4oU9VchaWYOlwDglJZVHWMpTT3Q=
k8s.io/cri-api v0.27.1/go.mod h1:+Ts/AVYbIo04S86XbTD73UPp/DkTiYxtsFeOFEu32L0=
k8s.io/gengo v0.0.0-20200413195148-3a45101e95ac/go.mod h1:ezvh/TsK7cY6rbqRK0oQQ8IAqLxYwwyPxAX1Pzy0ii0=
k8s.io/gengo v0.0.0-20200428234225-8167cfdcfc14/go.mod h1:ezvh/TsK7cY6rbqRK0oQQ8IAqLxYwwyPxAX1Pzy0ii0=
-k8s.io/gengo v0.0.0-20201113003025-83324d819ded/go.mod h1:FiNAH4ZV3gBg2Kwh89tzAEV2be7d5xI0vBa/VySYy3E=
-k8s.io/gengo v0.0.0-20220913193501-391367153a38 h1:yGN2TZt9XIl5wrcYaFtVMqzP2GIzX5gIcOObCZCuDeA=
-k8s.io/gengo v0.0.0-20220913193501-391367153a38/go.mod h1:FiNAH4ZV3gBg2Kwh89tzAEV2be7d5xI0vBa/VySYy3E=
+k8s.io/gengo v0.0.0-20230829151522-9cce18d56c01 h1:pWEwq4Asjm4vjW7vcsmijwBhOr1/shsbSYiWXmNGlks=
+k8s.io/gengo v0.0.0-20230829151522-9cce18d56c01/go.mod h1:FiNAH4ZV3gBg2Kwh89tzAEV2be7d5xI0vBa/VySYy3E=
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/v2 v2.0.0/go.mod h1:PBfzABfn139FHAV07az/IF9Wp1bkk3vpT2XSJ76fSDE=
k8s.io/klog/v2 v2.2.0/go.mod h1:Od+F08eJP+W3HUb4pSrPpgp9DGU4GzlpG/TmITuYh/Y=
-k8s.io/klog/v2 v2.4.0/go.mod h1:Od+F08eJP+W3HUb4pSrPpgp9DGU4GzlpG/TmITuYh/Y=
k8s.io/klog/v2 v2.100.1 h1:7WCHKK6K8fNhTqfBhISHQ97KrnJNFZMcQvKp7gP/tmg=
k8s.io/klog/v2 v2.100.1/go.mod h1:y1WjHnz7Dj687irZUWR/WLkLc5N1YHtjLdmgWjndZn0=
k8s.io/kube-aggregator v0.19.12 h1:OwyNUe/7/gxzEnaLd3sC9Yrpx0fZAERzvFslX5Qq5g8=
k8s.io/kube-openapi v0.0.0-20200805222855-6aeccd4b50c6/go.mod h1:UuqjUnNftUyPE5H64/qeyjQoUZhGpeFDVdxjTeEVN2o=
-k8s.io/kube-openapi v0.0.0-20201113171705-d219536bb9fd/go.mod h1:WOJ3KddDSol4tAGcJo0Tvi+dK12EcqSLqcWsryKMpfM=
-k8s.io/kube-openapi v0.0.0-20230308215209-15aac26d736a h1:gmovKNur38vgoWfGtP5QOGNOA7ki4n6qNYoFAgMlNvg=
-k8s.io/kube-openapi v0.0.0-20230308215209-15aac26d736a/go.mod h1:y5VtZWM9sHHc2ZodIH/6SHzXj+TPU5USoA8lcIeKEKY=
-k8s.io/kubectl v0.26.0 h1:xmrzoKR9CyNdzxBmXV7jW9Ln8WMrwRK6hGbbf69o4T0=
-k8s.io/kubectl v0.26.0/go.mod h1:eInP0b+U9XUJWSYeU9XZnTA+cVYuWyl3iYPGtru0qhQ=
+k8s.io/kube-openapi v0.0.0-20230918164632-68afd615200d h1:/CFeJBjBrZvHX09rObS2+2iEEDevMWYc1v3aIYAjIYI=
+k8s.io/kube-openapi v0.0.0-20230918164632-68afd615200d/go.mod h1:AsvuZPBlUDVuCdzJ87iajxtXuR9oktsTctW/R9wwouA=
+k8s.io/kubectl v0.28.2 h1:fOWOtU6S0smdNjG1PB9WFbqEIMlkzU5ahyHkc7ESHgM=
+k8s.io/kubectl v0.28.2/go.mod h1:6EQWTPySF1fn7yKoQZHYf9TPwIl2AygHEcJoxFekr64=
k8s.io/kubelet v0.26.1 h1:wQyCQYmLW6GN3v7gVTxnc3jAE4zMYDlzdF3FZV4rKas=
k8s.io/kubelet v0.26.1/go.mod h1:gFVZ1Ab4XdjtnYdVRATwGwku7FhTxo6LVEZwYoQaDT8=
-k8s.io/kubernetes v1.13.0/go.mod h1:ocZa8+6APFNC2tX1DZASIbocyYT5jHzqFVsY5aoB7Jk=
-k8s.io/metrics v0.26.3 h1:pHI8XtmBbGGdh7bL0s2C3v93fJfxyktHPAFsnRYnDTo=
-k8s.io/metrics v0.26.3/go.mod h1:NNnWARAAz+ZJTs75Z66fJTV7jHcVb3GtrlDszSIr3fE=
+k8s.io/metrics v0.28.2 h1:Z/oMk5SmiT/Ji1SaWOPfW2l9W831BLO9/XxDq9iS3ak=
+k8s.io/metrics v0.28.2/go.mod h1:QTIIdjMrq+KodO+rmp6R9Pr1LZO8kTArNtkWoQXw0sw=
k8s.io/utils v0.0.0-20200729134348-d5654de09c73/go.mod h1:jPW/WVKK9YHAvNhRxK0md/EJ228hCsBRufyofKtW8HA=
-k8s.io/utils v0.0.0-20201110183641-67b214c5f920/go.mod h1:jPW/WVKK9YHAvNhRxK0md/EJ228hCsBRufyofKtW8HA=
-k8s.io/utils v0.0.0-20230406110748-d93618cff8a2 h1:qY1Ad8PODbnymg2pRbkyMT/ylpTrCM8P2RJ0yroCyIk=
-k8s.io/utils v0.0.0-20230406110748-d93618cff8a2/go.mod h1:OLgZIPagt7ERELqWJFomSt595RzquPNLL48iOWgYOg0=
-oras.land/oras-go v1.2.2 h1:0E9tOHUfrNH7TCDk5KU0jVBEzCqbfdyuVfGmJ7ZeRPE=
-oras.land/oras-go v1.2.2/go.mod h1:Apa81sKoZPpP7CDciE006tSZ0x3Q3+dOoBcMZ/aNxvw=
+k8s.io/utils v0.0.0-20230726121419-3b25d923346b h1:sgn3ZU783SCgtaSJjpcVVlRqd6GSnlTLKgpAAttJvpI=
+k8s.io/utils v0.0.0-20230726121419-3b25d923346b/go.mod h1:OLgZIPagt7ERELqWJFomSt595RzquPNLL48iOWgYOg0=
+oras.land/oras-go v1.2.4 h1:djpBY2/2Cs1PV87GSJlxv4voajVOMZxqqtq9AB8YNvY=
+oras.land/oras-go v1.2.4/go.mod h1:DYcGfb3YF1nKjcezfX2SNlDAeQFKSXmf+qrFmrh4324=
periph.io/x/host/v3 v3.8.0 h1:T5ojZ2wvnZHGPS4h95N2ZpcCyHnsvH3YRZ1UUUiv5CQ=
periph.io/x/host/v3 v3.8.0/go.mod h1:rzOLH+2g9bhc6pWZrkCrmytD4igwQ2vxFw6Wn6ZOlLY=
rsc.io/binaryregexp v0.2.0/go.mod h1:qTv7/COck+e2FymRvadv62gMdZztPaShugOCi3I+8D8=
@@ -2930,21 +2596,17 @@ rsc.io/letsencrypt v0.0.3 h1:H7xDfhkaFFSYEJlKeq38RwX2jYcnTeHuDQyT+mMNMwM=
rsc.io/pdf v0.1.1/go.mod h1:n8OzWcQ6Sp37PL01nO98y4iUCRdTGarVfzxY20ICaU4=
rsc.io/quote/v3 v3.1.0/go.mod h1:yEA65RcK8LyAZtP9Kv3t0HmxON59tX3rD+tICJqUlj0=
rsc.io/sampler v1.3.0/go.mod h1:T1hPZKmBbMNahiBKFy5HrXp6adAjACjK9JXDnKaTXpA=
-sigs.k8s.io/apiserver-network-proxy/konnectivity-client v0.0.14/go.mod h1:LEScyzhFmoF5pso/YSeBstl57mOzx9xlU9n85RGrDQg=
-sigs.k8s.io/apiserver-network-proxy/konnectivity-client v0.0.15/go.mod h1:LEScyzhFmoF5pso/YSeBstl57mOzx9xlU9n85RGrDQg=
-sigs.k8s.io/controller-runtime v0.14.6 h1:oxstGVvXGNnMvY7TAESYk+lzr6S3V5VFxQ6d92KcwQA=
-sigs.k8s.io/controller-runtime v0.14.6/go.mod h1:WqIdsAY6JBsjfc/CqO0CORmNtoCtE4S6qbPc9s68h+0=
+sigs.k8s.io/controller-runtime v0.15.2 h1:9V7b7SDQSJ08IIsJ6CY1CE85Okhp87dyTMNDG0FS7f4=
+sigs.k8s.io/controller-runtime v0.15.2/go.mod h1:7ngYvp1MLT+9GeZ+6lH3LOlcHkp/+tzA/fmHa4iq9kk=
sigs.k8s.io/json v0.0.0-20221116044647-bc3834ca7abd h1:EDPBXCAspyGV4jQlpZSudPeMmr1bNJefnuqLsRAsHZo=
sigs.k8s.io/json v0.0.0-20221116044647-bc3834ca7abd/go.mod h1:B8JuhiUyNFVKdsE8h686QcCxMaH6HrOAZj4vswFpcB0=
-sigs.k8s.io/kustomize/api v0.12.1 h1:7YM7gW3kYBwtKvoY216ZzY+8hM+lV53LUayghNRJ0vM=
-sigs.k8s.io/kustomize/api v0.12.1/go.mod h1:y3JUhimkZkR6sbLNwfJHxvo1TCLwuwm14sCYnkH6S1s=
-sigs.k8s.io/kustomize/kustomize/v4 v4.5.7 h1:cDW6AVMl6t/SLuQaezMET8hgnadZGIAr8tUrxFVOrpg=
-sigs.k8s.io/kustomize/kustomize/v4 v4.5.7/go.mod h1:VSNKEH9D9d9bLiWEGbS6Xbg/Ih0tgQalmPvntzRxZ/Q=
-sigs.k8s.io/kustomize/kyaml v0.13.9 h1:Qz53EAaFFANyNgyOEJbT/yoIHygK40/ZcvU3rgry2Tk=
-sigs.k8s.io/kustomize/kyaml v0.13.9/go.mod h1:QsRbD0/KcU+wdk0/L0fIp2KLnohkVzs6fQ85/nOXac4=
+sigs.k8s.io/kustomize/api v0.13.5-0.20230601165947-6ce0bf390ce3 h1:XX3Ajgzov2RKUdc5jW3t5jwY7Bo7dcRm+tFxT+NfgY0=
+sigs.k8s.io/kustomize/api v0.13.5-0.20230601165947-6ce0bf390ce3/go.mod h1:9n16EZKMhXBNSiUC5kSdFQJkdH3zbxS/JoO619G1VAY=
+sigs.k8s.io/kustomize/kustomize/v5 v5.0.4-0.20230601165947-6ce0bf390ce3 h1:vq2TtoDcQomhy7OxXLUOzSbHMuMYq0Bjn93cDtJEdKw=
+sigs.k8s.io/kustomize/kustomize/v5 v5.0.4-0.20230601165947-6ce0bf390ce3/go.mod h1:/d88dHCvoy7d0AKFT0yytezSGZKjsZBVs9YTkBHSGFk=
+sigs.k8s.io/kustomize/kyaml v0.14.3 h1:WpabVAKZe2YEp/irTSHwD6bfjwZnTtSDewd2BVJGMZs=
+sigs.k8s.io/kustomize/kyaml v0.14.3/go.mod h1:npvh9epWysfQ689Rtt/U+dpOJDTBn8kUnF1O6VzvmZA=
sigs.k8s.io/structured-merge-diff/v4 v4.0.1/go.mod h1:bJZC9H9iH24zzfZ/41RGcq60oK1F7G282QMXDPYydCw=
-sigs.k8s.io/structured-merge-diff/v4 v4.0.2/go.mod h1:bJZC9H9iH24zzfZ/41RGcq60oK1F7G282QMXDPYydCw=
-sigs.k8s.io/structured-merge-diff/v4 v4.0.3/go.mod h1:bJZC9H9iH24zzfZ/41RGcq60oK1F7G282QMXDPYydCw=
sigs.k8s.io/structured-merge-diff/v4 v4.2.3 h1:PRbqxJClWWYMNV1dhaG4NsibJbArud9kFxnAMREiWFE=
sigs.k8s.io/structured-merge-diff/v4 v4.2.3/go.mod h1:qjx8mGObPmV2aSZepjQjbmb2ihdVs8cGKBraizNC69E=
sigs.k8s.io/yaml v1.1.0/go.mod h1:UJmg0vDUVViEyp3mgSv9WPwZCDxu4rQW1olrI1uml+o=
diff --git a/hack/client-sdk-gen.sh b/hack/client-sdk-gen.sh
index cfbd263275d..471751320fa 100755
--- a/hack/client-sdk-gen.sh
+++ b/hack/client-sdk-gen.sh
@@ -26,7 +26,10 @@ fi
CODE_GENERATOR_PATH=$(go list -f '{{.Dir}}' -m k8s.io/code-generator)
-GENERATORS="all" # deepcopy,defaulter,client,lister,informer or all
+# HACK: add exec permission to code generator scripts
+chmod u+x ${CODE_GENERATOR_PATH}/*.sh
+
+GENERATORS="client,informer,lister"
OUTPUT_PACKAGE="github.com/apecloud/kubeblocks/pkg/client"
APIS_PACKAGE="github.com/apecloud/kubeblocks/apis"
GROUP_VERSIONS="apps:v1alpha1 dataprotection:v1alpha1 extensions:v1alpha1 workloads:v1alpha1 storage:v1alpha1"
diff --git a/internal/cli/cloudprovider/k3d.go b/internal/cli/cloudprovider/k3d.go
index ee9d76d00da..29f681db68b 100644
--- a/internal/cli/cloudprovider/k3d.go
+++ b/internal/cli/cloudprovider/k3d.go
@@ -32,7 +32,7 @@ import (
"github.com/docker/go-connections/nat"
"github.com/k3d-io/k3d/v5/pkg/actions"
k3dClient "github.com/k3d-io/k3d/v5/pkg/client"
- config "github.com/k3d-io/k3d/v5/pkg/config/v1alpha4"
+ config "github.com/k3d-io/k3d/v5/pkg/config/v1alpha5"
l "github.com/k3d-io/k3d/v5/pkg/logger"
"github.com/k3d-io/k3d/v5/pkg/runtimes"
k3d "github.com/k3d-io/k3d/v5/pkg/types"
diff --git a/internal/cli/cluster/cluster.go b/internal/cli/cluster/cluster.go
index e147c0d8ac3..f5c9c9df89c 100644
--- a/internal/cli/cluster/cluster.go
+++ b/internal/cli/cluster/cluster.go
@@ -40,6 +40,7 @@ import (
"github.com/apecloud/kubeblocks/internal/cli/types"
"github.com/apecloud/kubeblocks/internal/cli/util"
"github.com/apecloud/kubeblocks/internal/constant"
+ dptypes "github.com/apecloud/kubeblocks/internal/dataprotection/types"
)
// ConditionsError cluster displays this status on list cmd when the status of ApplyResources or ProvisioningStarted condition is "False".
@@ -182,7 +183,7 @@ func (o *ObjectsGetter) Get() (*ClusterObjects, error) {
// filter back-up job pod
for _, pod := range objs.Pods.Items {
labels := pod.GetLabels()
- if labels[constant.DataProtectionLabelBackupNameKey] == "" {
+ if labels[dptypes.DataProtectionLabelBackupNameKey] == "" {
podList = append(podList, pod)
}
}
@@ -247,7 +248,7 @@ func (o *ObjectsGetter) Get() (*ClusterObjects, error) {
}
// filter backups with cluster uid for excluding same cluster name
for _, v := range backups {
- sourceClusterUID := v.Labels[constant.DataProtectionLabelClusterUIDKey]
+ sourceClusterUID := v.Labels[dptypes.DataProtectionLabelClusterUIDKey]
if sourceClusterUID == "" || sourceClusterUID == string(objs.Cluster.UID) {
objs.Backups = append(objs.Backups, v)
}
diff --git a/internal/cli/cluster/cluster_test.go b/internal/cli/cluster/cluster_test.go
index 7b777f25fae..055fdb036b3 100644
--- a/internal/cli/cluster/cluster_test.go
+++ b/internal/cli/cluster/cluster_test.go
@@ -26,9 +26,8 @@ import (
"k8s.io/apimachinery/pkg/runtime"
"k8s.io/client-go/kubernetes"
- dpv1alpha1 "github.com/apecloud/kubeblocks/apis/dataprotection/v1alpha1"
"github.com/apecloud/kubeblocks/internal/cli/testing"
- "github.com/apecloud/kubeblocks/internal/constant"
+ dptypes "github.com/apecloud/kubeblocks/internal/dataprotection/types"
)
var _ = Describe("cluster util", func() {
@@ -42,7 +41,7 @@ var _ = Describe("cluster util", func() {
baseObjsWithBackupPods := func() []runtime.Object {
podsWithBackup := testing.FakePods(4, testing.Namespace, testing.ClusterName)
labels := podsWithBackup.Items[0].GetLabels()
- labels[constant.DataProtectionLabelBackupNameKey] = string(dpv1alpha1.BackupTypeLogFile)
+ labels[dptypes.DataProtectionLabelBackupNameKey] = testing.BackupName
podsWithBackup.Items[0].SetLabels(labels)
return []runtime.Object{
podsWithBackup,
diff --git a/internal/cli/cmd/auth/authorize/callback_html/complete.html b/internal/cli/cmd/auth/authorize/authenticator/callback_html/complete.html
similarity index 100%
rename from internal/cli/cmd/auth/authorize/callback_html/complete.html
rename to internal/cli/cmd/auth/authorize/authenticator/callback_html/complete.html
diff --git a/internal/cli/cmd/auth/authorize/callback_html/error.html b/internal/cli/cmd/auth/authorize/authenticator/callback_html/error.html
similarity index 100%
rename from internal/cli/cmd/auth/authorize/callback_html/error.html
rename to internal/cli/cmd/auth/authorize/authenticator/callback_html/error.html
diff --git a/internal/cli/cmd/auth/authorize/authenticator/callback_listener.go b/internal/cli/cmd/auth/authorize/authenticator/callback_listener.go
index 23e8908cc05..672d678b02f 100644
--- a/internal/cli/cmd/auth/authorize/authenticator/callback_listener.go
+++ b/internal/cli/cmd/auth/authorize/authenticator/callback_listener.go
@@ -21,6 +21,7 @@ package authenticator
import (
"context"
+ "embed"
"errors"
"fmt"
"log"
@@ -28,11 +29,17 @@ import (
"os"
"os/signal"
"strings"
+
+ "github.com/leaanthony/debme"
+)
+
+var (
+ //go:embed callback_html/*
+ callbackHTML embed.FS
)
const (
ListenerAddress = "127.0.0.1"
- DIR = "./internal/cli/cmd/auth/authorize/callback_html/"
)
type HTTPServer interface {
@@ -114,18 +121,19 @@ func (c *CallbackService) awaitResponse(callbackResponse chan CallbackResponse,
})
}
-func writeHTML(w http.ResponseWriter, file string) {
- htmlContent, err := os.ReadFile(DIR + file)
+func writeHTML(w http.ResponseWriter, fileName string) {
+ tmplFs, _ := debme.FS(callbackHTML, "callback_html")
+ tmlBytes, err := tmplFs.ReadFile(fileName)
if err != nil {
http.Error(w, "Failed to read HTML file", http.StatusInternalServerError)
return
}
w.Header().Set("Content-Type", "text/html")
- write(w, string(htmlContent))
+ write(w, tmlBytes)
}
-func write(w http.ResponseWriter, msg string) {
- _, err := w.Write([]byte(msg))
+func write(w http.ResponseWriter, msg []byte) {
+ _, err := w.Write(msg)
if err != nil {
fmt.Println("Error writing response:", err)
}
diff --git a/internal/cli/cmd/auth/authorize/cached_provider.go b/internal/cli/cmd/auth/authorize/cached_provider.go
index e24ddca6052..f870d4d641a 100644
--- a/internal/cli/cmd/auth/authorize/cached_provider.go
+++ b/internal/cli/cmd/auth/authorize/cached_provider.go
@@ -38,7 +38,7 @@ const (
tokenFile = "token.json"
keyringKey = "token"
- keyringService = "kueblocks"
+ keyringService = "kubeblocks"
keyringLabel = "KUBEBLOCKS CLI"
fileMode = 0o600
diff --git a/internal/cli/cmd/auth/authorize/token_provider.go b/internal/cli/cmd/auth/authorize/token_provider.go
index f802c3de870..150545d7b9c 100644
--- a/internal/cli/cmd/auth/authorize/token_provider.go
+++ b/internal/cli/cmd/auth/authorize/token_provider.go
@@ -88,7 +88,6 @@ func (p *TokenProvider) Login(ctx context.Context) (*authenticator.UserInfoRespo
if err != nil {
return nil, "", errors.Wrap(err, "could not cache tokens")
}
- fmt.Println(tokenResult.IDToken)
return userInfo, tokenResult.IDToken, nil
}
diff --git a/internal/cli/cmd/backup/create.go b/internal/cli/cmd/backup/create.go
index 7f35944afd9..ab121cbe877 100644
--- a/internal/cli/cmd/backup/create.go
+++ b/internal/cli/cmd/backup/create.go
@@ -86,7 +86,7 @@ func newCreateCommand(f cmdutil.Factory, streams genericclioptions.IOStreams) *c
},
}
- cmd.Flags().StringVar(&o.BackupType, "type", "snapshot", "Backup type")
+ cmd.Flags().StringVar(&o.BackupMethod, "method", "snapshot", "Backup type")
cmd.Flags().StringVar(&clusterName, "cluster", "", "Cluster name")
cmd.Flags().StringVar(&o.BackupPolicy, "policy", "", "Backup policy name, this flag will be ignored when backup-type is snapshot")
util.RegisterClusterCompletionFunc(cmd, f)
diff --git a/internal/cli/cmd/backuprepo/create.go b/internal/cli/cmd/backuprepo/create.go
index 14d867d93ba..73db76b88c4 100644
--- a/internal/cli/cmd/backuprepo/create.go
+++ b/internal/cli/cmd/backuprepo/create.go
@@ -47,13 +47,13 @@ import (
"github.com/xeipuuv/gojsonschema"
"golang.org/x/exp/slices"
- dataprotectionv1alpha1 "github.com/apecloud/kubeblocks/apis/dataprotection/v1alpha1"
+ dpv1alpha1 "github.com/apecloud/kubeblocks/apis/dataprotection/v1alpha1"
storagev1alpha1 "github.com/apecloud/kubeblocks/apis/storage/v1alpha1"
"github.com/apecloud/kubeblocks/internal/cli/printer"
"github.com/apecloud/kubeblocks/internal/cli/types"
"github.com/apecloud/kubeblocks/internal/cli/util"
"github.com/apecloud/kubeblocks/internal/cli/util/flags"
- "github.com/apecloud/kubeblocks/internal/constant"
+ dptypes "github.com/apecloud/kubeblocks/internal/dataprotection/types"
)
const (
@@ -306,7 +306,7 @@ func (o *createOptions) validate(cmd *cobra.Command) error {
return err
}
for _, item := range list.Items {
- if item.GetAnnotations()[constant.DefaultBackupRepoAnnotationKey] == "true" {
+ if item.GetAnnotations()[dptypes.DefaultBackupRepoAnnotationKey] == "true" {
name := item.GetName()
return fmt.Errorf("there is already a default backup repo \"%s\","+
" please don't specify the --default flag,\n"+
@@ -346,12 +346,12 @@ func (o *createOptions) createCredentialSecret() (*corev1.Secret, error) {
}
func (o *createOptions) buildBackupRepoObject(secret *corev1.Secret) (*unstructured.Unstructured, error) {
- backupRepo := &dataprotectionv1alpha1.BackupRepo{
+ backupRepo := &dpv1alpha1.BackupRepo{
TypeMeta: metav1.TypeMeta{
APIVersion: fmt.Sprintf("%s/%s", types.DPAPIGroup, types.DPAPIVersion),
Kind: "BackupRepo",
},
- Spec: dataprotectionv1alpha1.BackupRepoSpec{
+ Spec: dpv1alpha1.BackupRepoSpec{
StorageProviderRef: o.storageProvider,
PVReclaimPolicy: corev1.PersistentVolumeReclaimPolicy(o.pvReclaimPolicy),
VolumeCapacity: resource.MustParse(o.volumeCapacity),
@@ -371,7 +371,7 @@ func (o *createOptions) buildBackupRepoObject(secret *corev1.Secret) (*unstructu
}
if o.isDefault {
backupRepo.Annotations = map[string]string{
- constant.DefaultBackupRepoAnnotationKey: "true",
+ dptypes.DefaultBackupRepoAnnotationKey: "true",
}
}
obj, err := runtime.DefaultUnstructuredConverter.ToUnstructured(backupRepo)
@@ -452,7 +452,7 @@ func registerFlagCompletionFunc(cmd *cobra.Command, f cmdutil.Factory) {
util.CheckErr(cmd.RegisterFlagCompletionFunc(
providerFlagName,
func(cmd *cobra.Command, args []string, toComplete string) ([]string, cobra.ShellCompDirective) {
- return utilcomp.CompGetResource(f, cmd, util.GVRToString(types.StorageProviderGVR()), toComplete), cobra.ShellCompDirectiveNoFileComp
+ return utilcomp.CompGetResource(f, util.GVRToString(types.StorageProviderGVR()), toComplete), cobra.ShellCompDirectiveNoFileComp
}))
util.CheckErr(cmd.RegisterFlagCompletionFunc(
"pv-reclaim-policy",
diff --git a/internal/cli/cmd/backuprepo/describe.go b/internal/cli/cmd/backuprepo/describe.go
index 87edd2764d1..dba6d0d0566 100644
--- a/internal/cli/cmd/backuprepo/describe.go
+++ b/internal/cli/cmd/backuprepo/describe.go
@@ -34,7 +34,7 @@ import (
cmdutil "k8s.io/kubectl/pkg/cmd/util"
"k8s.io/kubectl/pkg/util/templates"
- dataprotectionv1alpha1 "github.com/apecloud/kubeblocks/apis/dataprotection/v1alpha1"
+ dpv1alpha1 "github.com/apecloud/kubeblocks/apis/dataprotection/v1alpha1"
"github.com/apecloud/kubeblocks/internal/cli/printer"
"github.com/apecloud/kubeblocks/internal/cli/types"
"github.com/apecloud/kubeblocks/internal/cli/util"
@@ -111,7 +111,7 @@ func (o *describeBackupRepoOptions) run() error {
if err != nil {
return err
}
- backupRepo := &dataprotectionv1alpha1.BackupRepo{}
+ backupRepo := &dpv1alpha1.BackupRepo{}
if err = runtime.DefaultUnstructuredConverter.FromUnstructured(backupRepoObj.Object, backupRepo); err != nil {
return err
}
@@ -123,7 +123,7 @@ func (o *describeBackupRepoOptions) run() error {
return nil
}
-func (o *describeBackupRepoOptions) printBackupRepo(backupRepo *dataprotectionv1alpha1.BackupRepo) error {
+func (o *describeBackupRepoOptions) printBackupRepo(backupRepo *dpv1alpha1.BackupRepo) error {
printer.PrintLine("Summary:")
printer.PrintPairStringToLine("Name", backupRepo.Name)
printer.PrintPairStringToLine("Provider", backupRepo.Spec.StorageProviderRef)
@@ -153,7 +153,7 @@ func (o *describeBackupRepoOptions) printBackupRepo(backupRepo *dataprotectionv1
return nil
}
-func countBackupNumsAndSize(dynamic dynamic.Interface, backupRepo *dataprotectionv1alpha1.BackupRepo) (int, string, error) {
+func countBackupNumsAndSize(dynamic dynamic.Interface, backupRepo *dpv1alpha1.BackupRepo) (int, string, error) {
var size uint64
count := 0
@@ -166,12 +166,12 @@ func countBackupNumsAndSize(dynamic dynamic.Interface, backupRepo *dataprotectio
count = len(backupList.Items)
for _, obj := range backupList.Items {
- backup := &dataprotectionv1alpha1.Backup{}
+ backup := &dpv1alpha1.Backup{}
if err = runtime.DefaultUnstructuredConverter.FromUnstructured(obj.Object, backup); err != nil {
return count, humanize.Bytes(size), err
}
// if backup doesn't complete, we don't count it's size
- if backup.Status.Phase != dataprotectionv1alpha1.BackupCompleted {
+ if backup.Status.Phase != dpv1alpha1.BackupPhaseCompleted {
continue
}
backupSize, err := humanize.ParseBytes(backup.Status.TotalSize)
diff --git a/internal/cli/cmd/bench/bench.go b/internal/cli/cmd/bench/bench.go
index 51c46b6377a..73f0e5be5fe 100644
--- a/internal/cli/cmd/bench/bench.go
+++ b/internal/cli/cmd/bench/bench.go
@@ -129,6 +129,10 @@ func (o *BenchBaseOptions) BaseValidate() error {
return fmt.Errorf("port is required")
}
+ if err := validateBenchmarkExist(o.factory, o.IOStreams, o.name); err != nil {
+ return err
+ }
+
return nil
}
@@ -461,6 +465,9 @@ func validateBenchmarkExist(factory cmdutil.Factory, streams genericclioptions.I
bench.Print = false
result, err := bench.Run()
if err != nil {
+ if strings.Contains(err.Error(), "the server doesn't have a resource type") {
+ return fmt.Errorf("kubebench is not installed, please run `kbcli addon enable kubebench` to install it")
+ }
return err
}
diff --git a/internal/cli/cmd/bench/pgbench.go b/internal/cli/cmd/bench/pgbench.go
index 259a0b8b71b..eacc348d864 100644
--- a/internal/cli/cmd/bench/pgbench.go
+++ b/internal/cli/cmd/bench/pgbench.go
@@ -190,10 +190,6 @@ func (o *PgBenchOptions) Validate() error {
return fmt.Errorf("database is required")
}
- if err := validateBenchmarkExist(o.factory, o.IOStreams, o.name); err != nil {
- return err
- }
-
return nil
}
diff --git a/internal/cli/cmd/bench/sysbench.go b/internal/cli/cmd/bench/sysbench.go
index 29887d562b8..7c310eee601 100644
--- a/internal/cli/cmd/bench/sysbench.go
+++ b/internal/cli/cmd/bench/sysbench.go
@@ -206,10 +206,6 @@ func (o *SysBenchOptions) Validate() error {
return fmt.Errorf("database is required")
}
- if err := validateBenchmarkExist(o.factory, o.IOStreams, o.name); err != nil {
- return err
- }
-
if len(o.Type) == 0 {
return fmt.Errorf("type is required")
}
diff --git a/internal/cli/cmd/bench/ycsb.go b/internal/cli/cmd/bench/ycsb.go
index 35feb7d2a04..c182e84759a 100644
--- a/internal/cli/cmd/bench/ycsb.go
+++ b/internal/cli/cmd/bench/ycsb.go
@@ -193,10 +193,6 @@ func (o *YcsbOptions) Validate() error {
return fmt.Errorf("driver %s is not supported", o.Driver)
}
- if err := validateBenchmarkExist(o.factory, o.IOStreams, o.name); err != nil {
- return err
- }
-
if o.RecordCount < 0 {
return fmt.Errorf("record count should be positive")
}
diff --git a/internal/cli/cmd/builder/template/component_wrapper.go b/internal/cli/cmd/builder/template/component_wrapper.go
index dbda2085e23..7b1df33f7cb 100644
--- a/internal/cli/cmd/builder/template/component_wrapper.go
+++ b/internal/cli/cmd/builder/template/component_wrapper.go
@@ -328,10 +328,7 @@ func generateComponentObjects(w *templateRenderWorkflow, ctx intctrlutil.Request
if err != nil {
return nil, nil, err
}
- secret, err := factory.BuildConnCredential(w.clusterDefObj, cluster, component.GetSynthesizedComponent())
- if err != nil {
- return nil, nil, err
- }
+ secret := factory.BuildConnCredential(w.clusterDefObj, cluster, component.GetSynthesizedComponent())
cli.AppendMockObjects(secret)
if err = component.Create(ctx, cli); err != nil {
return nil, nil, err
diff --git a/internal/cli/cmd/builder/template/helm_helper.go b/internal/cli/cmd/builder/template/helm_helper.go
index 50d3b0f3841..5a1ba317a15 100644
--- a/internal/cli/cmd/builder/template/helm_helper.go
+++ b/internal/cli/cmd/builder/template/helm_helper.go
@@ -31,7 +31,7 @@ import (
"sigs.k8s.io/controller-runtime/pkg/client"
appsv1alpha1 "github.com/apecloud/kubeblocks/apis/apps/v1alpha1"
- dataprotectionv1alpha1 "github.com/apecloud/kubeblocks/apis/dataprotection/v1alpha1"
+ dpv1alpha1 "github.com/apecloud/kubeblocks/apis/dataprotection/v1alpha1"
"github.com/apecloud/kubeblocks/internal/generics"
)
@@ -120,8 +120,8 @@ func createObjectsFromYaml(yamlBytes []byte) ([]client.Object, error) {
objects = append(objects, CreateTypedObjectFromYamlByte(doc, generics.ClusterVersionSignature))
case kindFromResource(appsv1alpha1.BackupPolicyTemplate{}):
objects = append(objects, CreateTypedObjectFromYamlByte(doc, generics.BackupPolicyTemplateSignature))
- case kindFromResource(dataprotectionv1alpha1.BackupTool{}):
- objects = append(objects, CreateTypedObjectFromYamlByte(doc, generics.BackupToolSignature))
+ case kindFromResource(dpv1alpha1.ActionSet{}):
+ objects = append(objects, CreateTypedObjectFromYamlByte(doc, generics.ActionSetSignature))
}
}
return objects, nil
diff --git a/internal/cli/cmd/builder/template/k8s_resource.go b/internal/cli/cmd/builder/template/k8s_resource.go
index 94d26e7007f..b0898bd29a9 100644
--- a/internal/cli/cmd/builder/template/k8s_resource.go
+++ b/internal/cli/cmd/builder/template/k8s_resource.go
@@ -32,8 +32,8 @@ import (
type MatchResourceFunc func(object client.Object) bool
-func CustomizedObjFromYaml[T generics.Object, PT generics.PObject[T], L generics.ObjList[T]](filePath string, signature func(T, L)) (PT, error) {
- objList, err := CustomizedObjectListFromYaml[T, PT, L](filePath, signature)
+func CustomizedObjFromYaml[T generics.Object, PT generics.PObject[T], L generics.ObjList[T], PL generics.PObjList[T, L]](filePath string, signature func(T, PT, L, PL)) (PT, error) {
+ objList, err := CustomizedObjectListFromYaml[T, PT, L, PL](filePath, signature)
if err != nil {
return nil, err
}
@@ -43,7 +43,7 @@ func CustomizedObjFromYaml[T generics.Object, PT generics.PObject[T], L generics
return objList[0], nil
}
-func CustomizedObjectListFromYaml[T generics.Object, PT generics.PObject[T], L generics.ObjList[T]](yamlfile string, signature func(T, L)) ([]PT, error) {
+func CustomizedObjectListFromYaml[T generics.Object, PT generics.PObject[T], L generics.ObjList[T], PL generics.PObjList[T, L]](yamlfile string, signature func(T, PT, L, PL)) ([]PT, error) {
objBytes, err := os.ReadFile(yamlfile)
if err != nil {
return nil, err
@@ -58,12 +58,12 @@ func CustomizedObjectListFromYaml[T generics.Object, PT generics.PObject[T], L g
if err != nil {
return nil, err
}
- objList = append(objList, CreateTypedObjectFromYamlByte[T, PT, L](doc, signature))
+ objList = append(objList, CreateTypedObjectFromYamlByte[T, PT, L, PL](doc, signature))
}
return objList, nil
}
-func CreateTypedObjectFromYamlByte[T generics.Object, PT generics.PObject[T], L generics.ObjList[T]](yamlBytes []byte, _ func(T, L)) PT {
+func CreateTypedObjectFromYamlByte[T generics.Object, PT generics.PObject[T], L generics.ObjList[T], PL generics.PObjList[T, L]](yamlBytes []byte, _ func(T, PT, L, PL)) PT {
var obj PT
if err := yaml.Unmarshal(yamlBytes, &obj); err != nil {
return nil
@@ -71,7 +71,7 @@ func CreateTypedObjectFromYamlByte[T generics.Object, PT generics.PObject[T], L
return obj
}
-func GetTypedResourceObjectBySignature[T generics.Object, PT generics.PObject[T], L generics.ObjList[T]](objects []client.Object, _ func(T, L), matchers ...MatchResourceFunc) PT {
+func GetTypedResourceObjectBySignature[T generics.Object, PT generics.PObject[T], L generics.ObjList[T], PL generics.PObjList[T, L]](objects []client.Object, _ func(T, PT, L, PL), matchers ...MatchResourceFunc) PT {
for _, object := range objects {
obj, ok := object.(PT)
if !ok {
diff --git a/internal/cli/cmd/builder/template/mock_client.go b/internal/cli/cmd/builder/template/mock_client.go
index ab596e4599a..470f78168a3 100644
--- a/internal/cli/cmd/builder/template/mock_client.go
+++ b/internal/cli/cmd/builder/template/mock_client.go
@@ -26,6 +26,7 @@ import (
apierrors "k8s.io/apimachinery/pkg/api/errors"
"k8s.io/apimachinery/pkg/api/meta"
"k8s.io/apimachinery/pkg/runtime"
+ "k8s.io/apimachinery/pkg/runtime/schema"
"sigs.k8s.io/controller-runtime/pkg/client"
cfgcore "github.com/apecloud/kubeblocks/internal/configuration/core"
@@ -154,3 +155,11 @@ func (m *mockClient) Scheme() *runtime.Scheme {
func (m *mockClient) RESTMapper() meta.RESTMapper {
panic("implement me")
}
+
+func (m *mockClient) GroupVersionKindFor(obj runtime.Object) (schema.GroupVersionKind, error) {
+ panic("implement me")
+}
+
+func (m *mockClient) IsObjectNamespaced(obj runtime.Object) (bool, error) {
+ panic("implement me")
+}
diff --git a/internal/cli/cmd/class/create.go b/internal/cli/cmd/class/create.go
index 863d347b6af..20d2ec86701 100644
--- a/internal/cli/cmd/class/create.go
+++ b/internal/cli/cmd/class/create.go
@@ -269,7 +269,7 @@ func registerFlagCompletionFunc(cmd *cobra.Command, f cmdutil.Factory) {
util.CheckErr(cmd.RegisterFlagCompletionFunc(
"cluster-definition",
func(cmd *cobra.Command, args []string, toComplete string) ([]string, cobra.ShellCompDirective) {
- return utilcomp.CompGetResource(f, cmd, util.GVRToString(types.ClusterDefGVR()), toComplete), cobra.ShellCompDirectiveNoFileComp
+ return utilcomp.CompGetResource(f, util.GVRToString(types.ClusterDefGVR()), toComplete), cobra.ShellCompDirectiveNoFileComp
}))
util.CheckErr(cmd.RegisterFlagCompletionFunc(
"type",
diff --git a/internal/cli/cmd/cli.go b/internal/cli/cmd/cli.go
index c3985733c57..8f136673817 100644
--- a/internal/cli/cmd/cli.go
+++ b/internal/cli/cmd/cli.go
@@ -119,7 +119,7 @@ func NewDefaultCliCmd() *cobra.Command {
case "help", cobra.ShellCompRequestCmd, cobra.ShellCompNoDescRequestCmd:
// Don't search for a plugin
default:
- if err := kccmd.HandlePluginCommand(pluginHandler, cmdPathPieces); err != nil {
+ if err := kccmd.HandlePluginCommand(pluginHandler, cmdPathPieces, true); err != nil {
fmt.Fprintf(os.Stderr, "Error: %v\n", err)
os.Exit(1)
}
@@ -260,7 +260,7 @@ func registerCompletionFuncForGlobalFlags(cmd *cobra.Command, f cmdutil.Factory)
cmdutil.CheckErr(cmd.RegisterFlagCompletionFunc(
"namespace",
func(cmd *cobra.Command, args []string, toComplete string) ([]string, cobra.ShellCompDirective) {
- return utilcomp.CompGetResource(f, cmd, "namespace", toComplete), cobra.ShellCompDirectiveNoFileComp
+ return utilcomp.CompGetResource(f, "namespace", toComplete), cobra.ShellCompDirectiveNoFileComp
}))
cmdutil.CheckErr(cmd.RegisterFlagCompletionFunc(
"context",
diff --git a/internal/cli/cmd/cluster/config_util_test.go b/internal/cli/cmd/cluster/config_util_test.go
index b239e906994..05c76c604f2 100644
--- a/internal/cli/cmd/cluster/config_util_test.go
+++ b/internal/cli/cmd/cluster/config_util_test.go
@@ -53,7 +53,7 @@ func NewFakeOperationsOptions(ns, cName string, opsType appsv1alpha1.OpsType, ob
types.ClusterGVR(): types.KindCluster + "List",
types.ConfigConstraintGVR(): types.KindConfigConstraint + "List",
types.BackupGVR(): types.KindBackup + "List",
- types.RestoreJobGVR(): types.KindRestoreJob + "List",
+ types.RestoreGVR(): types.KindRestore + "List",
types.OpsGVR(): types.KindOps + "List",
}
baseOptions.Dynamic = dynamicfakeclient.NewSimpleDynamicClientWithCustomListKinds(scheme.Scheme, listMapping, objs...)
diff --git a/internal/cli/cmd/cluster/create.go b/internal/cli/cmd/cluster/create.go
index 44d17d23a1e..78e1291d667 100755
--- a/internal/cli/cmd/cluster/create.go
+++ b/internal/cli/cmd/cluster/create.go
@@ -53,7 +53,7 @@ import (
"k8s.io/kubectl/pkg/util/templates"
appsv1alpha1 "github.com/apecloud/kubeblocks/apis/apps/v1alpha1"
- dataprotectionv1alpha1 "github.com/apecloud/kubeblocks/apis/dataprotection/v1alpha1"
+ dpv1alpha1 "github.com/apecloud/kubeblocks/apis/dataprotection/v1alpha1"
"github.com/apecloud/kubeblocks/internal/class"
"github.com/apecloud/kubeblocks/internal/cli/cluster"
"github.com/apecloud/kubeblocks/internal/cli/create"
@@ -227,9 +227,9 @@ type CreateOptions struct {
RBACEnabled bool `json:"-"`
Storages []string `json:"-"`
// backup name to restore in creation
- Backup string `json:"backup,omitempty"`
- RestoreTime string `json:"restoreTime,omitempty"`
- SourceCluster string `json:"sourceCluster,omitempty"`
+ Backup string `json:"backup,omitempty"`
+ RestoreTime string `json:"restoreTime,omitempty"`
+ RestoreManagementPolicy string `json:"-"`
// backup config
BackupConfig *appsv1alpha1.ClusterBackup `json:"backupConfig,omitempty"`
@@ -262,7 +262,7 @@ func NewCreateCmd(f cmdutil.Factory, streams genericclioptions.IOStreams) *cobra
cmd.Flags().StringArrayVar(&o.Storages, "pvc", []string{}, "Set the cluster detail persistent volume claim, each '--pvc' corresponds to a component, and will override the simple configurations about storage by --set (e.g. --pvc type=mysql,name=data,mode=ReadWriteOnce,size=20Gi --pvc type=mysql,name=log,mode=ReadWriteOnce,size=1Gi)")
cmd.Flags().StringVar(&o.Backup, "backup", "", "Set a source backup to restore data")
cmd.Flags().StringVar(&o.RestoreTime, "restore-to-time", "", "Set a time for point in time recovery")
- cmd.Flags().StringVar(&o.SourceCluster, "source-cluster", "", "Set a source cluster for point in time recovery")
+ cmd.Flags().StringVar(&o.RestoreManagementPolicy, "volume-restore-policy", "Parallel", "the volume claim restore policy, supported values: [Serial, Parallel]")
cmd.Flags().BoolVar(&o.RBACEnabled, "rbac-enabled", false, "Specify whether rbac resources will be created by kbcli, otherwise KubeBlocks server will try to create rbac resources")
cmd.PersistentFlags().BoolVar(&o.EditBeforeCreate, "edit", o.EditBeforeCreate, "Edit the API resource before creating")
cmd.PersistentFlags().StringVar(&o.DryRun, "dry-run", "none", `Must be "client", or "server". If with client strategy, only print the object that would be sent, and no data is actually sent. If with server strategy, submit the server-side request, but no data is persistent.`)
@@ -308,7 +308,7 @@ func setMonitor(monitoringInterval uint8, components []map[string]interface{}) {
}
}
-func getRestoreFromBackupAnnotation(backup *dataprotectionv1alpha1.Backup, compSpecsCount int, firstCompName string) (string, error) {
+func getRestoreFromBackupAnnotation(backup *dpv1alpha1.Backup, managementPolicy string, compSpecsCount int, firstCompName string, restoreTime string) (string, error) {
componentName := backup.Labels[constant.KBAppComponentLabelKey]
if len(componentName) == 0 {
if compSpecsCount != 1 {
@@ -316,11 +316,19 @@ func getRestoreFromBackupAnnotation(backup *dataprotectionv1alpha1.Backup, compS
}
componentName = firstCompName
}
- restoreFromBackupAnnotation := fmt.Sprintf(`{"%s":"%s"}`, componentName, backup.Name)
+ backupNameString := fmt.Sprintf(`"%s":"%s"`, constant.BackupNameKeyForRestore, backup.Name)
+ backupNamespaceString := fmt.Sprintf(`"%s":"%s"`, constant.BackupNamespaceKeyForRestore, backup.Namespace)
+ managementPolicyString := fmt.Sprintf(`"%s":"%s"`, constant.VolumeManagementPolicyKeyForRestore, managementPolicy)
+ var restoreTimeString string
+ if restoreTime != "" {
+ restoreTimeString = fmt.Sprintf(`",%s":"%s"`, constant.RestoreTimeKeyForRestore, restoreTime)
+ }
+
+ restoreFromBackupAnnotation := fmt.Sprintf(`{"%s":{%s,%s,%s%s}}`, componentName, backupNameString, backupNamespaceString, managementPolicyString, restoreTimeString)
return restoreFromBackupAnnotation, nil
}
-func getSourceClusterFromBackup(backup *dataprotectionv1alpha1.Backup) (*appsv1alpha1.Cluster, error) {
+func getSourceClusterFromBackup(backup *dpv1alpha1.Backup) (*appsv1alpha1.Cluster, error) {
sourceCluster := &appsv1alpha1.Cluster{}
sourceClusterJSON := backup.Annotations[constant.ClusterSnapshotAnnotationKey]
if err := json.Unmarshal([]byte(sourceClusterJSON), sourceCluster); err != nil {
@@ -330,43 +338,21 @@ func getSourceClusterFromBackup(backup *dataprotectionv1alpha1.Backup) (*appsv1a
return sourceCluster, nil
}
-func getBackupObjectFromRestoreArgs(o *CreateOptions, backup *dataprotectionv1alpha1.Backup) error {
- if o.Backup != "" {
- if err := cluster.GetK8SClientObject(o.Dynamic, backup, types.BackupGVR(), o.Namespace, o.Backup); err != nil {
- return err
- }
- } else if o.RestoreTime != "" {
- createRestoreOptions := CreateRestoreOptions{
- SourceCluster: o.SourceCluster,
- RestoreTimeStr: o.RestoreTime,
- }
- createRestoreOptions.Dynamic = o.Dynamic
- createRestoreOptions.Namespace = o.Namespace
- if err := createRestoreOptions.validateRestoreTime(); err != nil {
- return err
- }
- objs, err := o.Dynamic.Resource(types.BackupGVR()).Namespace(o.Namespace).
- List(context.TODO(), metav1.ListOptions{
- LabelSelector: fmt.Sprintf("%s=%s",
- constant.AppInstanceLabelKey, o.SourceCluster),
- })
- if err != nil {
- return err
- }
- if len(objs.Items) == 0 {
- return fmt.Errorf("can not found any backup to restore time")
- }
-
- return runtime.DefaultUnstructuredConverter.FromUnstructured(objs.Items[0].UnstructuredContent(), backup)
+func getBackupObjectFromRestoreArgs(o *CreateOptions, backup *dpv1alpha1.Backup) error {
+ if o.Backup == "" {
+ return nil
+ }
+ if err := cluster.GetK8SClientObject(o.Dynamic, backup, types.BackupGVR(), o.Namespace, o.Backup); err != nil {
+ return err
}
return nil
}
func fillClusterInfoFromBackup(o *CreateOptions, cls **appsv1alpha1.Cluster) error {
- if o.Backup == "" && o.RestoreTime == "" && o.SourceCluster == "" {
+ if o.Backup == "" {
return nil
}
- backup := &dataprotectionv1alpha1.Backup{}
+ backup := &dpv1alpha1.Backup{}
if err := getBackupObjectFromRestoreArgs(o, backup); err != nil {
return err
}
@@ -374,12 +360,6 @@ func fillClusterInfoFromBackup(o *CreateOptions, cls **appsv1alpha1.Cluster) err
if err != nil {
return err
}
- // HACK/TODO: apecloud-mysql pitr only support one replica for PITR.
- if backupCluster.Spec.ClusterDefRef == apeCloudMysql && o.RestoreTime != "" {
- for _, c := range backupCluster.Spec.ComponentSpecs {
- c.Replicas = 1
- }
- }
curCluster := *cls
if curCluster == nil {
curCluster = backupCluster
@@ -402,54 +382,53 @@ func fillClusterInfoFromBackup(o *CreateOptions, cls **appsv1alpha1.Cluster) err
return nil
}
+func formatRestoreTimeAndValidate(restoreTimeStr string, continuousBackup *dpv1alpha1.Backup) (string, error) {
+ if restoreTimeStr == "" {
+ return restoreTimeStr, nil
+ }
+ restoreTime, err := util.TimeParse(restoreTimeStr, time.Second)
+ if err != nil {
+ // retry to parse time with RFC3339 format.
+ var errRFC error
+ restoreTime, errRFC = time.Parse(time.RFC3339, restoreTimeStr)
+ if errRFC != nil {
+ // if retry failure, report the error
+ return restoreTimeStr, err
+ }
+ }
+ restoreTimeStr = restoreTime.Format(time.RFC3339)
+ // TODO: check with Recoverable time
+ if !isTimeInRange(restoreTime, continuousBackup.Status.TimeRange.Start.Time, continuousBackup.Status.TimeRange.End.Time) {
+ return restoreTimeStr, fmt.Errorf("restore-to-time is out of time range, you can view the recoverable time: \n"+
+ "\tkbcli cluster describe %s -n %s", continuousBackup.Labels[constant.AppInstanceLabelKey], continuousBackup.Namespace)
+ }
+ return restoreTimeStr, nil
+}
+
func setBackup(o *CreateOptions, components []map[string]interface{}) error {
backupName := o.Backup
if len(backupName) == 0 || len(components) == 0 {
return nil
}
- backup := &dataprotectionv1alpha1.Backup{}
+ backup := &dpv1alpha1.Backup{}
if err := cluster.GetK8SClientObject(o.Dynamic, backup, types.BackupGVR(), o.Namespace, backupName); err != nil {
return err
}
- if backup.Status.Phase != dataprotectionv1alpha1.BackupCompleted {
+ if backup.Status.Phase != dpv1alpha1.BackupPhaseCompleted {
return fmt.Errorf(`backup "%s" is not completed`, backup.Name)
}
- restoreAnnotation, err := getRestoreFromBackupAnnotation(backup, len(components), components[0]["name"].(string))
+ restoreTimeStr, err := formatRestoreTimeAndValidate(o.RestoreTime, backup)
if err != nil {
return err
}
- if o.Annotations == nil {
- o.Annotations = map[string]string{}
- }
- o.Annotations[constant.RestoreFromBackUpAnnotationKey] = restoreAnnotation
- return nil
-}
-
-func setRestoreTime(o *CreateOptions, components []map[string]interface{}) error {
- if o.RestoreTime == "" || o.SourceCluster == "" {
- return nil
- }
-
- // HACK/TODO: apecloud-mysql pitr only support one replica for PITR.
- if o.ClusterDefRef == apeCloudMysql {
- for _, c := range components {
- if c["replicas"].(int64) > 1 {
- return fmt.Errorf("apecloud-mysql only support one replica for point-in-time recovery")
- }
- }
+ restoreAnnotation, err := getRestoreFromBackupAnnotation(backup, o.RestoreManagementPolicy, len(components), components[0]["name"].(string), restoreTimeStr)
+ if err != nil {
+ return err
}
-
if o.Annotations == nil {
o.Annotations = map[string]string{}
}
- restoreTime, err := util.TimeParse(o.RestoreTime, time.Second)
- if err != nil {
- return err
- }
- // TODO: hack implement for multi-component cluster, how to elegantly implement pitr for multi-component cluster?
- o.Annotations[constant.RestoreFromTimeAnnotationKey] = fmt.Sprintf(`{"%s":"%s"}`, components[0]["name"], restoreTime.Format(time.RFC3339))
- o.Annotations[constant.RestoreFromSrcClusterAnnotationKey] = o.SourceCluster
-
+ o.Annotations[constant.RestoreFromBackupAnnotationKey] = restoreAnnotation
return nil
}
@@ -548,9 +527,6 @@ func (o *CreateOptions) Complete() error {
if err = setBackup(o, components); err != nil {
return err
}
- if err = setRestoreTime(o, components); err != nil {
- return err
- }
o.ComponentSpecs = components
// TolerationsRaw looks like `["key=engineType,value=mongo,operator=Equal,effect=NoSchedule"]` after parsing by cmd
@@ -869,7 +845,7 @@ func registerFlagCompletionFunc(cmd *cobra.Command, f cmdutil.Factory) {
util.CheckErr(cmd.RegisterFlagCompletionFunc(
"cluster-definition",
func(cmd *cobra.Command, args []string, toComplete string) ([]string, cobra.ShellCompDirective) {
- return utilcomp.CompGetResource(f, cmd, util.GVRToString(types.ClusterDefGVR()), toComplete), cobra.ShellCompDirectiveNoFileComp
+ return utilcomp.CompGetResource(f, util.GVRToString(types.ClusterDefGVR()), toComplete), cobra.ShellCompDirectiveNoFileComp
}))
util.CheckErr(cmd.RegisterFlagCompletionFunc(
"cluster-version",
@@ -877,7 +853,7 @@ func registerFlagCompletionFunc(cmd *cobra.Command, f cmdutil.Factory) {
var clusterVersion []string
clusterDefinition, err := cmd.Flags().GetString("cluster-definition")
if clusterDefinition == "" || err != nil {
- clusterVersion = utilcomp.CompGetResource(f, cmd, util.GVRToString(types.ClusterVersionGVR()), toComplete)
+ clusterVersion = utilcomp.CompGetResource(f, util.GVRToString(types.ClusterVersionGVR()), toComplete)
} else {
label := fmt.Sprintf("%s=%s", constant.ClusterDefLabelKey, clusterDefinition)
clusterVersion = util.CompGetResourceWithLabels(f, cmd, util.GVRToString(types.ClusterVersionGVR()), []string{label}, toComplete)
@@ -1404,9 +1380,7 @@ func (o *CreateOptions) buildAnnotation(cls *appsv1alpha1.Cluster) {
func (o *CreateOptions) buildBackupConfig(cls *appsv1alpha1.Cluster) error {
// set default backup config
- o.BackupConfig = &appsv1alpha1.ClusterBackup{
- Method: dataprotectionv1alpha1.BackupMethodSnapshot,
- }
+ o.BackupConfig = &appsv1alpha1.ClusterBackup{}
// if the cls.Backup isn't nil, use the backup config in cluster
if cls != nil && cls.Spec.Backup != nil {
@@ -1427,9 +1401,9 @@ func (o *CreateOptions) buildBackupConfig(cls *appsv1alpha1.Cluster) error {
case "backup-enabled":
o.BackupConfig.Enabled = &o.BackupEnabled
case "backup-retention-period":
- o.BackupConfig.RetentionPeriod = &o.BackupRetentionPeriod
+ o.BackupConfig.RetentionPeriod = dpv1alpha1.RetentionPeriod(o.BackupRetentionPeriod)
case "backup-method":
- o.BackupConfig.Method = dataprotectionv1alpha1.BackupMethod(o.BackupMethod)
+ o.BackupConfig.Method = o.BackupMethod
case "backup-cron-expression":
if _, err := cron.ParseStandard(o.BackupCronExpression); err != nil {
return fmt.Errorf("invalid cron expression: %s, please see https://en.wikipedia.org/wiki/Cron", o.BackupCronExpression)
diff --git a/internal/cli/cmd/cluster/create_test.go b/internal/cli/cmd/cluster/create_test.go
index 8edea4d46d9..4f4854e6445 100644
--- a/internal/cli/cmd/cluster/create_test.go
+++ b/internal/cli/cmd/cluster/create_test.go
@@ -425,20 +425,6 @@ var _ = Describe("create", func() {
Expect(setBackup(o, components)).Should(Succeed())
})
- It("set restoreTime", func() {
- o := &CreateOptions{}
- o.Namespace = testing.Namespace
- o.RestoreTime = "Jun 16,2023 18:57:01 UTC+0800"
- o.SourceCluster = testing.ClusterName
- components := []map[string]interface{}{
- {
- "name": testing.ClusterName,
- },
- }
- By("test setRestoreTime")
- Expect(setRestoreTime(o, components)).Should(Succeed())
- })
-
It("test fillClusterMetadataFromBackup", func() {
baseBackupName := "test-backup"
logBackupName := "test-logfile-backup"
@@ -452,32 +438,30 @@ var _ = Describe("create", func() {
o.Dynamic = dynamic
o.Namespace = testing.Namespace
o.RestoreTime = "Jun 16,2023 18:57:01 UTC+0800"
+ o.Backup = logBackupName
backupLogTime, _ := util.TimeParse(o.RestoreTime, time.Second)
- o.SourceCluster = clusterName
buildBackupLogTime := func(d time.Duration) string {
return backupLogTime.Add(d).Format(time.RFC3339)
}
- buildManifests := func(startTime, stopTime string) map[string]any {
+ buildTimeRange := func(startTime, stopTime string) map[string]any {
return map[string]any{
- "backupLog": map[string]any{
- "startTime": startTime,
- "stopTime": stopTime,
- },
+ "start": startTime,
+ "end": stopTime,
}
}
- mockBackupInfo(dynamic, baseBackupName, clusterName, buildManifests(buildBackupLogTime(-30*time.Second), buildBackupLogTime(-10*time.Second)), "snapshot")
- mockBackupInfo(dynamic, logBackupName, clusterName, buildManifests(buildBackupLogTime(-1*time.Minute), buildBackupLogTime(time.Minute)), "logfile")
+ mockBackupInfo(dynamic, baseBackupName, clusterName, buildTimeRange(buildBackupLogTime(-30*time.Second), buildBackupLogTime(-10*time.Second)), "snapshot")
+ mockBackupInfo(dynamic, logBackupName, clusterName, buildTimeRange(buildBackupLogTime(-1*time.Minute), buildBackupLogTime(time.Minute)), "logfile")
By("fill cluster from backup success")
Expect(fillClusterInfoFromBackup(o, &cluster)).Should(Succeed())
Expect(cluster.Spec.ClusterDefRef).Should(Equal(testing.ClusterDefName))
Expect(cluster.Spec.ClusterVersionRef).Should(Equal(testing.ClusterVersionName))
- By("fill cluster definition does not matched")
+ By("fill cluster definition does not match")
o.ClusterDefRef = "test-not-match-cluster-definition"
Expect(fillClusterInfoFromBackup(o, &cluster)).Should(HaveOccurred())
o.ClusterDefRef = ""
- By("fill cluster version does not matched")
+ By("fill cluster version does not match")
o.ClusterVersionRef = "test-not-match-cluster-version"
Expect(fillClusterInfoFromBackup(o, &cluster)).Should(HaveOccurred())
})
@@ -492,9 +476,9 @@ var _ = Describe("create", func() {
By("test backup is with snapshot method")
o.BackupMethod = "snapshot"
- Expect(o.Cmd.Flags().Set("backup", "snapshot")).To(Succeed())
+ Expect(o.Cmd.Flags().Set("backup-method", "snapshot")).To(Succeed())
Expect(o.buildBackupConfig(cluster)).To(Succeed())
- Expect(string(o.BackupConfig.Method)).Should(Equal("snapshot"))
+ Expect(o.BackupConfig.Method).Should(Equal("snapshot"))
By("test backup is with wrong cron expression")
o.BackupCronExpression = "wrong-cron-expression"
diff --git a/internal/cli/cmd/cluster/create_util.go b/internal/cli/cmd/cluster/create_util.go
index e579ed03a76..ccc9cbf7780 100644
--- a/internal/cli/cmd/cluster/create_util.go
+++ b/internal/cli/cmd/cluster/create_util.go
@@ -115,7 +115,7 @@ func registerFlagCompFunc(cmd *cobra.Command, f cmdutil.Factory, c *cluster.Char
label := fmt.Sprintf("%s=%s", constant.ClusterDefLabelKey, c.ClusterDef)
versions = util.CompGetResourceWithLabels(f, cmd, util.GVRToString(types.ClusterVersionGVR()), []string{label}, toComplete)
} else {
- versions = utilcomp.CompGetResource(f, cmd, util.GVRToString(types.ClusterVersionGVR()), toComplete)
+ versions = utilcomp.CompGetResource(f, util.GVRToString(types.ClusterVersionGVR()), toComplete)
}
return versions, cobra.ShellCompDirectiveNoFileComp
})
diff --git a/internal/cli/cmd/cluster/dataprotection.go b/internal/cli/cmd/cluster/dataprotection.go
index 9bd60008ca1..2936247de0e 100644
--- a/internal/cli/cmd/cluster/dataprotection.go
+++ b/internal/cli/cmd/cluster/dataprotection.go
@@ -25,11 +25,9 @@ import (
"fmt"
"reflect"
"sort"
- "strconv"
"strings"
"time"
- "github.com/dapr/kit/cron"
"github.com/pkg/errors"
"github.com/spf13/cobra"
batchv1 "k8s.io/api/batch/v1"
@@ -61,6 +59,7 @@ import (
"github.com/apecloud/kubeblocks/internal/cli/types"
"github.com/apecloud/kubeblocks/internal/cli/util"
"github.com/apecloud/kubeblocks/internal/constant"
+ dptypes "github.com/apecloud/kubeblocks/internal/dataprotection/types"
)
var (
@@ -121,7 +120,7 @@ var (
const annotationTrueValue = "true"
type CreateBackupOptions struct {
- BackupType string `json:"backupType"`
+ BackupMethod string `json:"backupMethod"`
BackupName string `json:"backupName"`
Role string `json:"role,omitempty"`
BackupPolicy string `json:"backupPolicy"`
@@ -173,8 +172,10 @@ func (o *CreateBackupOptions) Validate() error {
return err
}
}
- if o.BackupType == string(dpv1alpha1.BackupTypeLogFile) {
- return fmt.Errorf(`can not create logfile backup, you can create it by enabling spec.schedule.logfile in BackupPolicy "%s"`, o.BackupPolicy)
+ if o.BackupMethod == "" {
+ // TODO(ldm): if backup policy only has one backup method, use it as default
+ // backup method.
+ return fmt.Errorf("missing backup method")
}
// TODO: check if pvc exists
return nil
@@ -212,7 +213,7 @@ func (o *CreateBackupOptions) getDefaultBackupPolicy() (string, error) {
}
var defaultBackupPolicies []unstructured.Unstructured
for _, obj := range objs.Items {
- if obj.GetAnnotations()[constant.DefaultBackupPolicyAnnotationKey] == annotationTrueValue {
+ if obj.GetAnnotations()[dptypes.DefaultBackupPolicyAnnotationKey] == annotationTrueValue {
defaultBackupPolicies = append(defaultBackupPolicies, obj)
}
}
@@ -258,7 +259,7 @@ func NewCreateBackupCmd(f cmdutil.Factory, streams genericclioptions.IOStreams)
},
}
- cmd.Flags().StringVar(&o.BackupType, "type", "snapshot", "Backup type")
+ cmd.Flags().StringVar(&o.BackupMethod, "method", "", "Backup method that defined in backup policy")
cmd.Flags().StringVar(&o.BackupName, "name", "", "Backup name")
cmd.Flags().StringVar(&o.BackupPolicy, "policy", "", "Backup policy name, this flag will be ignored when backup-type is snapshot")
@@ -302,28 +303,23 @@ func PrintBackupList(o ListBackupOptions) error {
// sort the unstructured objects with the creationTimestamp in positive order
sort.Sort(unstructuredList(backupList.Items))
tbl := printer.NewTablePrinter(o.Out)
- tbl.SetHeader("NAME", "NAMESPACE", "SOURCE-CLUSTER", "TYPE", "STATUS", "TOTAL-SIZE", "DURATION", "CREATE-TIME", "COMPLETION-TIME", "EXPIRATION")
+ tbl.SetHeader("NAME", "NAMESPACE", "SOURCE-CLUSTER", "METHOD", "STATUS", "TOTAL-SIZE", "DURATION", "CREATE-TIME", "COMPLETION-TIME", "EXPIRATION")
for _, obj := range backupList.Items {
backup := &dpv1alpha1.Backup{}
if err = runtime.DefaultUnstructuredConverter.FromUnstructured(obj.Object, backup); err != nil {
return err
}
- sourceCluster := backup.Status.SourceCluster
- if sourceCluster == "" {
- sourceCluster = backup.Labels[constant.AppInstanceLabelKey]
- }
+ // TODO(ldm): find cluster from backup policy target spec.
+ sourceCluster := backup.Labels[constant.AppInstanceLabelKey]
durationStr := ""
if backup.Status.Duration != nil {
durationStr = duration.HumanDuration(backup.Status.Duration.Duration)
}
statusString := string(backup.Status.Phase)
- if backup.Status.Phase == dpv1alpha1.BackupRunning && backup.Status.AvailableReplicas != nil {
- statusString = fmt.Sprintf("%s(AvailablePods: %d)", statusString, *backup.Status.AvailableReplicas)
- }
if len(o.Names) > 0 && !backupNameMap[backup.Name] {
continue
}
- tbl.AddRow(backup.Name, backup.Namespace, sourceCluster, backup.Spec.BackupType, statusString, backup.Status.TotalSize,
+ tbl.AddRow(backup.Name, backup.Namespace, sourceCluster, backup.Spec.BackupMethod, statusString, backup.Status.TotalSize,
durationStr, util.TimeFormat(&backup.CreationTimestamp), util.TimeFormat(backup.Status.CompletionTimestamp),
util.TimeFormat(backup.Status.Expiration))
}
@@ -419,9 +415,9 @@ type CreateRestoreOptions struct {
Backup string `json:"backup,omitempty"`
// point in time recovery args
- RestoreTime *time.Time `json:"restoreTime,omitempty"`
- RestoreTimeStr string `json:"restoreTimeStr,omitempty"`
- SourceCluster string `json:"sourceCluster,omitempty"`
+ RestoreTime *time.Time `json:"restoreTime,omitempty"`
+ RestoreTimeStr string `json:"restoreTimeStr,omitempty"`
+ RestoreManagementPolicy string `json:"volumeRestorePolicy,omitempty"`
create.CreateOptions `json:"-"`
}
@@ -443,8 +439,6 @@ func (o *CreateRestoreOptions) getClusterObject(backup *dpv1alpha1.Backup) (*app
func (o *CreateRestoreOptions) Run() error {
if o.Backup != "" {
return o.runRestoreFromBackup()
- } else if o.RestoreTime != nil {
- return o.runPITR()
}
return nil
}
@@ -455,25 +449,30 @@ func (o *CreateRestoreOptions) runRestoreFromBackup() error {
if err := cluster.GetK8SClientObject(o.Dynamic, backup, types.BackupGVR(), o.Namespace, o.Backup); err != nil {
return err
}
- if backup.Status.Phase != dpv1alpha1.BackupCompleted {
+ if backup.Status.Phase != dpv1alpha1.BackupPhaseCompleted {
return errors.Errorf(`backup "%s" is not completed.`, backup.Name)
}
if len(backup.Labels[constant.AppInstanceLabelKey]) == 0 {
return errors.Errorf(`missing source cluster in backup "%s", "app.kubernetes.io/instance" is empty in labels.`, o.Backup)
}
+
+ restoreTimeStr, err := formatRestoreTimeAndValidate(o.RestoreTimeStr, backup)
+ if err != nil {
+ return err
+ }
// get the cluster object and set the annotation for restore
clusterObj, err := o.getClusterObject(backup)
if err != nil {
return err
}
- restoreAnnotation, err := getRestoreFromBackupAnnotation(backup, len(clusterObj.Spec.ComponentSpecs), clusterObj.Spec.ComponentSpecs[0].Name)
+ restoreAnnotation, err := getRestoreFromBackupAnnotation(backup, o.RestoreManagementPolicy, len(clusterObj.Spec.ComponentSpecs), clusterObj.Spec.ComponentSpecs[0].Name, restoreTimeStr)
if err != nil {
return err
}
clusterObj.ObjectMeta = metav1.ObjectMeta{
Namespace: clusterObj.Namespace,
Name: o.Name,
- Annotations: map[string]string{constant.RestoreFromBackUpAnnotationKey: restoreAnnotation},
+ Annotations: map[string]string{constant.RestoreFromBackupAnnotationKey: restoreAnnotation},
}
return o.createCluster(clusterObj)
}
@@ -500,103 +499,13 @@ func (o *CreateRestoreOptions) createCluster(cluster *appsv1alpha1.Cluster) erro
return nil
}
-func (o *CreateRestoreOptions) runPITR() error {
- objs, err := o.Dynamic.Resource(types.BackupGVR()).Namespace(o.Namespace).
- List(context.TODO(), metav1.ListOptions{
- LabelSelector: fmt.Sprintf("%s=%s",
- constant.AppInstanceLabelKey, o.SourceCluster),
- })
- if err != nil {
- return err
- }
- backup := &dpv1alpha1.Backup{}
-
- // no need to check items len because it is validated by o.validateRestoreTime().
- if err := runtime.DefaultUnstructuredConverter.FromUnstructured(objs.Items[0].Object, backup); err != nil {
- return err
- }
- compName := backup.Labels[constant.KBAppComponentLabelKey]
- if compName == "" {
- return fmt.Errorf(`component name label %s is missing in backup "%s"`, constant.KBAppComponentLabelKey, backup.Name)
- }
- // TODO: use opsRequest to create cluster.
- // get the cluster object and set the annotation for restore
- clusterObj, err := o.getClusterObject(backup)
- if err != nil {
- return err
- }
- // TODO: hack implement for multi-component cluster, how to elegantly implement pitr for multi-component cluster?
- clusterObj.ObjectMeta = metav1.ObjectMeta{
- Namespace: clusterObj.Namespace,
- Name: o.Name,
- Annotations: map[string]string{
- constant.RestoreFromTimeAnnotationKey: fmt.Sprintf(`{"%s":"%s"}`, compName, o.RestoreTime.Format(time.RFC3339)),
- constant.RestoreFromSrcClusterAnnotationKey: o.SourceCluster,
- },
- }
- return o.createCluster(clusterObj)
-}
-
func isTimeInRange(t time.Time, start time.Time, end time.Time) bool {
return !t.Before(start) && !t.After(end)
}
-func (o *CreateRestoreOptions) validateRestoreTime() error {
- if o.RestoreTimeStr == "" && o.SourceCluster == "" {
- return nil
- }
- if o.RestoreTimeStr != "" && o.SourceCluster == "" {
- return fmt.Errorf("--source-cluster must be specified if specified --restore-to-time")
- }
- restoreTime, err := util.TimeParse(o.RestoreTimeStr, time.Second)
- if err != nil {
- // retry to parse time with RFC3339 format.
- var errRFC error
- restoreTime, errRFC = time.Parse(time.RFC3339, o.RestoreTimeStr)
- if errRFC != nil {
- // if retry failure, report the error
- return err
- }
- }
- o.RestoreTime = &restoreTime
- objs, err := o.Dynamic.Resource(types.BackupGVR()).Namespace(o.Namespace).
- List(context.TODO(), metav1.ListOptions{
- LabelSelector: fmt.Sprintf("%s=%s",
- constant.AppInstanceLabelKey, o.SourceCluster),
- })
- if err != nil {
- return err
- }
- backupMap := map[string][]dpv1alpha1.Backup{}
- for _, i := range objs.Items {
- obj := dpv1alpha1.Backup{}
- if err = runtime.DefaultUnstructuredConverter.FromUnstructured(i.Object, &obj); err != nil {
- return err
- }
- uid := obj.Labels[constant.DataProtectionLabelClusterUIDKey]
- if backupMap[uid] == nil {
- backupMap[uid] = make([]dpv1alpha1.Backup, 0)
- }
- backupMap[uid] = append(backupMap[uid], obj)
- }
- for _, v := range backupMap {
- recoverableTime := dpv1alpha1.GetRecoverableTimeRange(v)
- for _, i := range recoverableTime {
- if isTimeInRange(restoreTime, i.StartTime.Time, i.StopTime.Time) {
- return nil
- }
- }
- }
- return fmt.Errorf("restore-to-time is out of time range, you can view the recoverable time: \n"+
- "\tkbcli cluster describe %s -n %s", o.SourceCluster, o.Namespace)
-}
-
func (o *CreateRestoreOptions) Validate() error {
- if o.Backup == "" && o.RestoreTimeStr == "" {
- return fmt.Errorf("must be specified one of the --backup or --restore-to-time")
- }
- if err := o.validateRestoreTime(); err != nil {
- return err
+ if o.Backup == "" {
+ return fmt.Errorf("must be specified one of the --backup ")
}
if o.Name == "" {
@@ -634,7 +543,7 @@ func NewCreateRestoreCmd(f cmdutil.Factory, streams genericclioptions.IOStreams)
}
cmd.Flags().StringVar(&o.Backup, "backup", "", "Backup name")
cmd.Flags().StringVar(&o.RestoreTimeStr, "restore-to-time", "", "point in time recovery(PITR)")
- cmd.Flags().StringVar(&o.SourceCluster, "source-cluster", "", "source cluster name")
+ cmd.Flags().StringVar(&o.RestoreManagementPolicy, "volume-restore-policy", "Parallel", "the volume claim restore policy, supported values: [Serial, Parallel]")
return cmd
}
@@ -688,7 +597,7 @@ func printBackupPolicyList(o list.ListOptions) error {
tbl := printer.NewTablePrinter(o.Out)
tbl.SetHeader("NAME", "NAMESPACE", "DEFAULT", "CLUSTER", "CREATE-TIME", "STATUS")
for _, obj := range backupPolicyList.Items {
- defaultPolicy, ok := obj.GetAnnotations()[constant.DefaultBackupPolicyAnnotationKey]
+ defaultPolicy, ok := obj.GetAnnotations()[dptypes.DefaultBackupPolicyAnnotationKey]
backupPolicy := &dpv1alpha1.BackupPolicy{}
if err = runtime.DefaultUnstructuredConverter.FromUnstructured(obj.Object, backupPolicy); err != nil {
return err
@@ -766,28 +675,7 @@ func (o *editBackupPolicyOptions) complete(args []string) error {
if o.dynamic, err = o.Factory.DynamicClient(); err != nil {
return err
}
- updateSchedulePolicyEnable := func(schedulePolicy *dpv1alpha1.SchedulePolicy, targetVal string) error {
- if schedulePolicy != nil {
- enable, err := strconv.ParseBool(targetVal)
- if err != nil {
- return err
- }
- schedulePolicy.Enable = enable
- }
- return nil
- }
- updateSchedulePolicyCronExpression := func(schedulePolicy *dpv1alpha1.SchedulePolicy, targetVal string) error {
- if targetVal != "" {
- if _, err = cron.ParseStandard(targetVal); err != nil {
- return err
- }
- }
- if schedulePolicy != nil {
- schedulePolicy.CronExpression = targetVal
- }
- return nil
- }
- updateRepoName := func(commonPolicy *dpv1alpha1.CommonBackupPolicy, targetVal string) error {
+ updateRepoName := func(backupPolicy *dpv1alpha1.BackupPolicy, targetVal string) error {
// check if the backup repo exists
if targetVal != "" {
_, err := o.dynamic.Resource(types.BackupRepoGVR()).Get(context.Background(), targetVal, metav1.GetOptions{})
@@ -795,117 +683,22 @@ func (o *editBackupPolicyOptions) complete(args []string) error {
return err
}
}
- if commonPolicy != nil {
+ if backupPolicy != nil {
if targetVal != "" {
- commonPolicy.BackupRepoName = &targetVal
+ backupPolicy.Spec.BackupRepoName = &targetVal
} else {
- commonPolicy.BackupRepoName = nil
+ backupPolicy.Spec.BackupRepoName = nil
}
}
return nil
}
- updatePVCName := func(commonPolicy *dpv1alpha1.CommonBackupPolicy, targetVal string) error {
- if commonPolicy != nil {
- commonPolicy.PersistentVolumeClaim.Name = &targetVal
- }
- return nil
- }
- updatePVCStorageClass := func(commonPolicy *dpv1alpha1.CommonBackupPolicy, targetVal string) error {
- if commonPolicy != nil {
- commonPolicy.PersistentVolumeClaim.StorageClassName = &targetVal
- }
- return nil
- }
+
o.editContent = []editorRow{
{
- key: "retention.ttl",
- jsonpath: "retention.ttl",
- updateFunc: func(backupPolicy *dpv1alpha1.BackupPolicy, targetVal string) error {
- backupPolicy.Spec.Retention.TTL = &targetVal
- return nil
- },
- },
- {
- key: "schedule.datafile.enable",
- jsonpath: "schedule.datafile.enable",
- updateFunc: func(backupPolicy *dpv1alpha1.BackupPolicy, targetVal string) error {
- return updateSchedulePolicyEnable(backupPolicy.Spec.Schedule.Datafile, targetVal)
- },
- },
- {
- key: "schedule.datafile.cronExpression",
- jsonpath: "schedule.datafile.cronExpression",
- updateFunc: func(backupPolicy *dpv1alpha1.BackupPolicy, targetVal string) error {
- return updateSchedulePolicyCronExpression(backupPolicy.Spec.Schedule.Datafile, targetVal)
- },
- },
- {
- key: "schedule.snapshot.enable",
- jsonpath: "schedule.snapshot.enable",
- updateFunc: func(backupPolicy *dpv1alpha1.BackupPolicy, targetVal string) error {
- return updateSchedulePolicyEnable(backupPolicy.Spec.Schedule.Snapshot, targetVal)
- },
- },
- {
- key: "schedule.snapshot.cronExpression",
- jsonpath: "schedule.snapshot.cronExpression",
- updateFunc: func(backupPolicy *dpv1alpha1.BackupPolicy, targetVal string) error {
- return updateSchedulePolicyCronExpression(backupPolicy.Spec.Schedule.Snapshot, targetVal)
- },
- },
- {
- key: "schedule.logfile.enable",
- jsonpath: "schedule.logfile.enable",
- updateFunc: func(backupPolicy *dpv1alpha1.BackupPolicy, targetVal string) error {
- return updateSchedulePolicyEnable(backupPolicy.Spec.Schedule.Logfile, targetVal)
- },
- },
- {
- key: "schedule.logfile.cronExpression",
- jsonpath: "schedule.logfile.cronExpression", updateFunc: func(backupPolicy *dpv1alpha1.BackupPolicy, targetVal string) error {
- return updateSchedulePolicyCronExpression(backupPolicy.Spec.Schedule.Logfile, targetVal)
- },
- },
- {
- key: "datafile.pvc.name",
- jsonpath: "datafile.persistentVolumeClaim.name",
- updateFunc: func(backupPolicy *dpv1alpha1.BackupPolicy, targetVal string) error {
- return updatePVCName(backupPolicy.Spec.Datafile, targetVal)
- },
- },
- {
- key: "datafile.pvc.storageClassName",
- jsonpath: "datafile.persistentVolumeClaim.storageClassName",
- updateFunc: func(backupPolicy *dpv1alpha1.BackupPolicy, targetVal string) error {
- return updatePVCStorageClass(backupPolicy.Spec.Datafile, targetVal)
- },
- },
- {
- key: "datafile.backupRepoName",
- jsonpath: "datafile.backupRepoName",
- updateFunc: func(backupPolicy *dpv1alpha1.BackupPolicy, targetVal string) error {
- return updateRepoName(backupPolicy.Spec.Datafile, targetVal)
- },
- },
- {
- key: "logfile.pvc.name",
- jsonpath: "logfile.persistentVolumeClaim.name",
- updateFunc: func(backupPolicy *dpv1alpha1.BackupPolicy, targetVal string) error {
- return updatePVCName(backupPolicy.Spec.Logfile, targetVal)
- },
- },
- {
- key: "logfile.pvc.storageClassName",
- jsonpath: "logfile.persistentVolumeClaim.storageClassName",
- updateFunc: func(backupPolicy *dpv1alpha1.BackupPolicy, targetVal string) error {
- return updatePVCStorageClass(backupPolicy.Spec.Logfile, targetVal)
- },
- },
- {
- key: "logfile.backupRepoName",
- jsonpath: "logfile.backupRepoName",
+ key: "backupRepoName",
+ jsonpath: "backupRepoName",
updateFunc: func(backupPolicy *dpv1alpha1.BackupPolicy, targetVal string) error {
- return updateRepoName(backupPolicy.Spec.Logfile, targetVal)
+ return updateRepoName(backupPolicy, targetVal)
},
},
}
@@ -1097,23 +890,23 @@ func (o *DescribeBackupOptions) Run() error {
}
func (o *DescribeBackupOptions) printBackupObj(obj *dpv1alpha1.Backup) error {
+ targetCluster := obj.Labels[constant.AppInstanceLabelKey]
printer.PrintLineWithTabSeparator(
printer.NewPair("Name", obj.Name),
- printer.NewPair("Cluster", obj.Status.SourceCluster),
+ printer.NewPair("Cluster", targetCluster),
printer.NewPair("Namespace", obj.Namespace),
)
printer.PrintLine("\nSpec:")
- realPrintPairStringToLine("Type", string(obj.Spec.BackupType))
+ realPrintPairStringToLine("Method", obj.Spec.BackupMethod)
realPrintPairStringToLine("Policy Name", obj.Spec.BackupPolicyName)
printer.PrintLine("\nStatus:")
realPrintPairStringToLine("Phase", string(obj.Status.Phase))
realPrintPairStringToLine("Total Size", obj.Status.TotalSize)
- realPrintPairStringToLine("Backup Tool", obj.Status.BackupToolName)
- realPrintPairStringToLine("PVC Name", obj.Status.PersistentVolumeClaimName)
- if obj.Status.AvailableReplicas != nil {
- realPrintPairStringToLine("Available Replicas", string(*obj.Status.AvailableReplicas))
+ if obj.Status.BackupMethod != nil {
+ realPrintPairStringToLine("ActionSet Name", obj.Status.BackupMethod.ActionSetName)
}
+ realPrintPairStringToLine("PVC Name", obj.Status.PersistentVolumeClaimName)
if obj.Status.Duration != nil {
realPrintPairStringToLine("Duration", duration.HumanDuration(obj.Status.Duration.Duration))
}
@@ -1123,26 +916,20 @@ func (o *DescribeBackupOptions) printBackupObj(obj *dpv1alpha1.Backup) error {
// print failure reason, ignore error
_ = o.enhancePrintFailureReason(obj.Name, obj.Status.FailureReason)
- if obj.Status.Manifests != nil {
- printer.PrintLine("\nManifests:")
- realPrintPairStringToLine("Target", obj.Status.Manifests.Target)
- if obj.Status.Manifests.BackupLog != nil {
- realPrintPairStringToLine("Log Start Time", util.TimeFormat(obj.Status.Manifests.BackupLog.StartTime))
- realPrintPairStringToLine("Log Stop Time", util.TimeFormat(obj.Status.Manifests.BackupLog.StopTime))
- }
- if obj.Status.Manifests.BackupTool != nil {
- realPrintPairStringToLine("File Path", obj.Status.Manifests.BackupTool.FilePath)
- realPrintPairStringToLine("Volume Name", obj.Status.Manifests.BackupTool.VolumeName)
- realPrintPairStringToLine("Upload Total Size", obj.Status.Manifests.BackupTool.UploadTotalSize)
- realPrintPairStringToLine("Checksum", obj.Status.Manifests.BackupTool.Checksum)
- realPrintPairStringToLine("Checkpoint", obj.Status.Manifests.BackupTool.Checkpoint)
- }
- if obj.Status.Manifests.Snapshot != nil {
- realPrintPairStringToLine("Snapshot Name", obj.Status.Manifests.Snapshot.VolumeSnapshotName)
- realPrintPairStringToLine("Snapshot Content Name", obj.Status.Manifests.Snapshot.VolumeSnapshotContentName)
- }
- for k, v := range obj.Status.Manifests.UserContext {
- realPrintPairStringToLine(k, v)
+ realPrintPairStringToLine("Path", obj.Status.Path)
+
+ if obj.Status.TimeRange != nil {
+ realPrintPairStringToLine("Time Range Start", util.TimeFormat(obj.Status.TimeRange.Start))
+ realPrintPairStringToLine("Time Range End", util.TimeFormat(obj.Status.TimeRange.End))
+ }
+
+ if len(obj.Status.VolumeSnapshots) > 0 {
+ printer.PrintLine("\nVolume Snapshots:")
+ for _, v := range obj.Status.VolumeSnapshots {
+ realPrintPairStringToLine("Name", v.Name)
+ realPrintPairStringToLine("Content Name", v.ContentName)
+ realPrintPairStringToLine("Volume Name:", v.VolumeName)
+ realPrintPairStringToLine("Size", v.Size)
}
}
@@ -1173,7 +960,7 @@ func (o *DescribeBackupOptions) enhancePrintFailureReason(backupName, failureRea
ctx := context.Background()
// get the latest job log details.
labels := fmt.Sprintf("%s=%s",
- constant.DataProtectionLabelBackupNameKey, backupName,
+ dptypes.DataProtectionLabelBackupNameKey, backupName,
)
jobList, err := o.client.BatchV1().Jobs("").List(ctx, metav1.ListOptions{LabelSelector: labels})
if err != nil {
diff --git a/internal/cli/cmd/cluster/dataprotection_test.go b/internal/cli/cmd/cluster/dataprotection_test.go
index 98482a8d75a..0d808d0d3b1 100644
--- a/internal/cli/cmd/cluster/dataprotection_test.go
+++ b/internal/cli/cmd/cluster/dataprotection_test.go
@@ -35,8 +35,6 @@ import (
k8sapitypes "k8s.io/apimachinery/pkg/types"
"k8s.io/cli-runtime/pkg/genericclioptions"
"k8s.io/client-go/dynamic"
- "k8s.io/client-go/dynamic/fake"
- "k8s.io/client-go/kubernetes/scheme"
clientfake "k8s.io/client-go/rest/fake"
cmdtesting "k8s.io/kubectl/pkg/cmd/testing"
@@ -50,7 +48,6 @@ import (
"github.com/apecloud/kubeblocks/internal/cli/types"
"github.com/apecloud/kubeblocks/internal/cli/util"
"github.com/apecloud/kubeblocks/internal/constant"
- testapps "github.com/apecloud/kubeblocks/internal/testutil/apps"
)
var _ = Describe("DataProtection", func() {
@@ -121,29 +118,13 @@ var _ = Describe("DataProtection", func() {
By("test edit backup policy function")
o := editBackupPolicyOptions{Factory: tf, IOStreams: streams, GVR: types.BackupPolicyGVR()}
Expect(o.complete([]string{policyName})).Should(Succeed())
- o.values = []string{"schedule.datafile.enable=false", `schedule.datafile.cronExpression="0 17 * * *"`,
- "schedule.logfile.enable=false", `schedule.logfile.cronExpression="* */1 * * *"`,
- "schedule.snapshot.enable=false", `schedule.snapshot.cronExpression="0 17 * * *"`,
- "logfile.pvc.name=test1", "logfile.pvc.storageClassName=t1", "logfile.backupRepoName=repo",
- "datafile.pvc.name=test1", "datafile.pvc.storageClassName=t1", "datafile.backupRepoName=repo"}
- Expect(o.runEditBackupPolicy()).Should(Succeed())
-
- By("test unset backup repo")
- o.values = []string{"datafile.backupRepoName="}
+ o.values = []string{"backupRepoName=repo"}
Expect(o.runEditBackupPolicy()).Should(Succeed())
By("test backup repo not exists")
- o.values = []string{"datafile.backupRepoName=repo1"}
+ o.values = []string{"backupRepoName=repo1"}
Expect(o.runEditBackupPolicy()).Should(MatchError(ContainSubstring(`"repo1" not found`)))
- By("test invalid key")
- o.values = []string{"schedule.datafile.enable1=false"}
- Expect(o.runEditBackupPolicy().Error()).Should(ContainSubstring("invalid key: schedule.datafile.enable1"))
-
- By("test invalid value")
- o.values = []string{"schedule.datafile.enable=false="}
- Expect(o.runEditBackupPolicy().Error()).Should(ContainSubstring("invalid row"))
-
By("test with vim editor")
o.values = []string{}
o.isTest = true
@@ -177,6 +158,7 @@ var _ = Describe("DataProtection", func() {
By("test with one default backupPolicy")
initClient(defaultBackupPolicy)
o.Dynamic = tf.FakeDynamicClient
+ o.BackupMethod = testing.BackupMethodName
Expect(o.Validate()).Should(Succeed())
})
@@ -187,7 +169,8 @@ var _ = Describe("DataProtection", func() {
cmd := NewCreateBackupCmd(tf, streams)
Expect(cmd).ShouldNot(BeNil())
// must succeed otherwise exit 1 and make test fails
- _ = cmd.Flags().Set("backup-policy", defaultBackupPolicy.Name)
+ _ = cmd.Flags().Set("policy", defaultBackupPolicy.Name)
+ _ = cmd.Flags().Set("method", testing.BackupMethodName)
cmd.Run(cmd, []string{testing.ClusterName})
By("test with logfile type")
@@ -200,12 +183,11 @@ var _ = Describe("DataProtection", func() {
Name: testing.ClusterName,
},
BackupPolicy: defaultBackupPolicy.Name,
- BackupType: string(dpv1alpha1.BackupTypeLogFile),
+ BackupMethod: testing.BackupMethodName,
}
Expect(o.CompleteBackup()).Should(Succeed())
err := o.Validate()
- Expect(err).Should(HaveOccurred())
- Expect(err.Error()).Should(ContainSubstring("can not create logfile backup, you can create it by enabling spec.schedule.logfile in BackupPolicy"))
+ Expect(err).Should(Succeed())
})
})
@@ -248,16 +230,13 @@ var _ = Describe("DataProtection", func() {
backup1.Labels = map[string]string{
constant.AppInstanceLabelKey: "apecloud-mysql",
}
- AvailableReplicas := int32(1)
- backup1.Status.Phase = dpv1alpha1.BackupRunning
- backup1.Status.AvailableReplicas = &AvailableReplicas
+ backup1.Status.Phase = dpv1alpha1.BackupPhaseRunning
backup2 := testing.FakeBackup("test1")
backup2.Namespace = "backup"
tf.FakeDynamicClient = testing.FakeDynamicClient(backup1, backup2)
Expect(PrintBackupList(o)).Should(Succeed())
Expect(o.Out.(*bytes.Buffer).String()).Should(ContainSubstring("test1"))
Expect(o.Out.(*bytes.Buffer).String()).Should(ContainSubstring("apecloud-mysql"))
- Expect(o.Out.(*bytes.Buffer).String()).Should(ContainSubstring("(AvailablePods: 1)"))
By("test list all namespace")
o.Out.(*bytes.Buffer).Reset()
@@ -281,15 +260,13 @@ var _ = Describe("DataProtection", func() {
backupPolicy := testing.FakeBackupPolicy("backPolicy", clusterObj.Name)
pods := testing.FakePods(1, testing.Namespace, clusterName)
- tf.FakeDynamicClient = fake.NewSimpleDynamicClient(
- scheme.Scheme, &secrets.Items[0], &pods.Items[0], clusterObj, backupPolicy)
- tf.FakeDynamicClient = fake.NewSimpleDynamicClient(
- scheme.Scheme, &secrets.Items[0], &pods.Items[0], clusterDef, clusterObj, backupPolicy)
+ tf.FakeDynamicClient = testing.FakeDynamicClient(&secrets.Items[0],
+ &pods.Items[0], clusterDef, clusterObj, backupPolicy)
tf.Client = &clientfake.RESTClient{}
// create backup
cmd := NewCreateBackupCmd(tf, streams)
Expect(cmd).ShouldNot(BeNil())
- _ = cmd.Flags().Set("type", "snapshot")
+ _ = cmd.Flags().Set("method", testing.BackupMethodName)
_ = cmd.Flags().Set("name", backupName)
cmd.Run(nil, []string{clusterName})
@@ -315,77 +292,77 @@ var _ = Describe("DataProtection", func() {
cmdRestore.Run(nil, []string{newClusterName + "-with-nil-affinity"})
})
- It("restore-to-time", func() {
- timestamp := time.Now().Format("20060102150405")
- backupName := "backup-test-" + timestamp
- backupName1 := backupName + "1"
- clusterName := "source-cluster-" + timestamp
- secrets := testing.FakeSecrets(testing.Namespace, clusterName)
- clusterDef := testing.FakeClusterDef()
- cluster := testing.FakeCluster(clusterName, testing.Namespace)
- clusterDefLabel := map[string]string{
- constant.ClusterDefLabelKey: clusterDef.Name,
- }
- cluster.SetLabels(clusterDefLabel)
- backupPolicy := testing.FakeBackupPolicy("backPolicy", cluster.Name)
- backupTypeMeta := testing.FakeBackup("backup-none").TypeMeta
- backupLabels := map[string]string{
- constant.AppInstanceLabelKey: clusterName,
- constant.KBAppComponentLabelKey: "test",
- constant.DataProtectionLabelClusterUIDKey: string(cluster.UID),
- }
- now := metav1.Now()
- baseBackup := testapps.NewBackupFactory(testing.Namespace, "backup-base").
- SetBackupType(dpv1alpha1.BackupTypeSnapshot).
- SetBackLog(now.Add(-time.Minute), now.Add(-time.Second)).
- SetLabels(backupLabels).GetObject()
- baseBackup.TypeMeta = backupTypeMeta
- baseBackup.Status.Phase = dpv1alpha1.BackupCompleted
- logfileBackup := testapps.NewBackupFactory(testing.Namespace, backupName).
- SetBackupType(dpv1alpha1.BackupTypeLogFile).
- SetBackLog(now.Add(-time.Minute), now.Add(time.Minute)).
- SetLabels(backupLabels).GetObject()
- logfileBackup.TypeMeta = backupTypeMeta
-
- logfileBackup1 := testapps.NewBackupFactory(testing.Namespace, backupName1).
- SetBackupType(dpv1alpha1.BackupTypeLogFile).
- SetBackLog(now.Add(-time.Minute), now.Add(2*time.Minute)).GetObject()
- uid := string(cluster.UID)
- logfileBackup1.Labels = map[string]string{
- constant.AppInstanceLabelKey: clusterName,
- constant.KBAppComponentLabelKey: "test",
- constant.DataProtectionLabelClusterUIDKey: uid[:30] + "00",
- }
- logfileBackup1.TypeMeta = backupTypeMeta
-
- pods := testing.FakePods(1, testing.Namespace, clusterName)
- tf.FakeDynamicClient = fake.NewSimpleDynamicClient(
- scheme.Scheme, &secrets.Items[0], &pods.Items[0], cluster, backupPolicy, baseBackup, logfileBackup, logfileBackup1)
- tf.Client = &clientfake.RESTClient{}
-
- By("restore new cluster from source cluster which is not deleted")
- cmdRestore := NewCreateRestoreCmd(tf, streams)
- Expect(cmdRestore != nil).To(BeTrue())
- _ = cmdRestore.Flags().Set("restore-to-time", util.TimeFormatWithDuration(&now, time.Second))
- _ = cmdRestore.Flags().Set("source-cluster", clusterName)
- cmdRestore.Run(nil, []string{})
-
- // test with RFC3339 format
- _ = cmdRestore.Flags().Set("restore-to-time", now.Format(time.RFC3339))
- _ = cmdRestore.Flags().Set("source-cluster", clusterName)
- cmdRestore.Run(nil, []string{"new-cluster"})
-
- By("restore should be failed when backups belong to different source clusters")
- o := &CreateRestoreOptions{CreateOptions: create.CreateOptions{
- IOStreams: streams,
- Factory: tf,
- }}
- restoreTime := time.Now().Add(90 * time.Second)
- o.RestoreTimeStr = util.TimeFormatWithDuration(&metav1.Time{Time: restoreTime}, time.Second)
- o.SourceCluster = clusterName
- Expect(o.Complete()).Should(Succeed())
- Expect(o.validateRestoreTime().Error()).Should(ContainSubstring("restore-to-time is out of time range"))
- })
+ // It("restore-to-time", func() {
+ // timestamp := time.Now().Format("20060102150405")
+ // backupName := "backup-test-" + timestamp
+ // backupName1 := backupName + "1"
+ // clusterName := "source-cluster-" + timestamp
+ // secrets := testing.FakeSecrets(testing.Namespace, clusterName)
+ // clusterDef := testing.FakeClusterDef()
+ // cluster := testing.FakeCluster(clusterName, testing.Namespace)
+ // clusterDefLabel := map[string]string{
+ // constant.ClusterDefLabelKey: clusterDef.Name,
+ // }
+ // cluster.SetLabels(clusterDefLabel)
+ // backupPolicy := testing.FakeBackupPolicy("backPolicy", cluster.Name)
+ // backupTypeMeta := testing.FakeBackup("backup-none").TypeMeta
+ // backupLabels := map[string]string{
+ // constant.AppInstanceLabelKey: clusterName,
+ // constant.KBAppComponentLabelKey: "test",
+ // dptypes.DataProtectionLabelClusterUIDKey: string(cluster.UID),
+ // }
+ // now := metav1.Now()
+ // baseBackup := testapps.NewBackupFactory(testing.Namespace, "backup-base").
+ // SetBackupMethod(dpv1alpha1.BackupTypeSnapshot).
+ // SetBackupTimeRange(now.Add(-time.Minute), now.Add(-time.Second)).
+ // SetLabels(backupLabels).GetObject()
+ // baseBackup.TypeMeta = backupTypeMeta
+ // baseBackup.Status.Phase = dpv1alpha1.BackupPhaseCompleted
+ // logfileBackup := testapps.NewBackupFactory(testing.Namespace, backupName).
+ // SetBackupMethod(dpv1alpha1.BackupTypeLogFile).
+ // SetBackupTimeRange(now.Add(-time.Minute), now.Add(time.Minute)).
+ // SetLabels(backupLabels).GetObject()
+ // logfileBackup.TypeMeta = backupTypeMeta
+ //
+ // logfileBackup1 := testapps.NewBackupFactory(testing.Namespace, backupName1).
+ // SetBackupMethod(dpv1alpha1.BackupTypeLogFile).
+ // SetBackupTimeRange(now.Add(-time.Minute), now.Add(2*time.Minute)).GetObject()
+ // uid := string(cluster.UID)
+ // logfileBackup1.Labels = map[string]string{
+ // constant.AppInstanceLabelKey: clusterName,
+ // constant.KBAppComponentLabelKey: "test",
+ // constant.DataProtectionLabelClusterUIDKey: uid[:30] + "00",
+ // }
+ // logfileBackup1.TypeMeta = backupTypeMeta
+ //
+ // pods := testing.FakePods(1, testing.Namespace, clusterName)
+ // tf.FakeDynamicClient = fake.NewSimpleDynamicClient(
+ // scheme.Scheme, &secrets.Items[0], &pods.Items[0], cluster, backupPolicy, baseBackup, logfileBackup, logfileBackup1)
+ // tf.Client = &clientfake.RESTClient{}
+ //
+ // By("restore new cluster from source cluster which is not deleted")
+ // cmdRestore := NewCreateRestoreCmd(tf, streams)
+ // Expect(cmdRestore != nil).To(BeTrue())
+ // _ = cmdRestore.Flags().Set("restore-to-time", util.TimeFormatWithDuration(&now, time.Second))
+ // _ = cmdRestore.Flags().Set("source-cluster", clusterName)
+ // cmdRestore.Run(nil, []string{})
+ //
+ // // test with RFC3339 format
+ // _ = cmdRestore.Flags().Set("restore-to-time", now.Format(time.RFC3339))
+ // _ = cmdRestore.Flags().Set("source-cluster", clusterName)
+ // cmdRestore.Run(nil, []string{"new-cluster"})
+ //
+ // By("restore should be failed when backups belong to different source clusters")
+ // o := &CreateRestoreOptions{CreateOptions: create.CreateOptions{
+ // IOStreams: streams,
+ // Factory: tf,
+ // }}
+ // restoreTime := time.Now().Add(90 * time.Second)
+ // o.RestoreTimeStr = util.TimeFormatWithDuration(&metav1.Time{Time: restoreTime}, time.Second)
+ // o.SourceCluster = clusterName
+ // Expect(o.Complete()).Should(Succeed())
+ // Expect(o.validateRestoreTime().Error()).Should(ContainSubstring("restore-to-time is out of time range"))
+ // })
It("describe-backup", func() {
cmd := NewDescribeBackupCmd(tf, streams)
@@ -404,21 +381,12 @@ var _ = Describe("DataProtection", func() {
backupName := "test1"
backup1 := testing.FakeBackup(backupName)
args = append(args, backupName)
- availableReplicas := int32(1)
- backup1.Status.Phase = dpv1alpha1.BackupCompleted
+ backup1.Status.Phase = dpv1alpha1.BackupPhaseCompleted
logNow := metav1.Now()
backup1.Status.StartTimestamp = &logNow
backup1.Status.CompletionTimestamp = &logNow
backup1.Status.Expiration = &logNow
backup1.Status.Duration = &metav1.Duration{Duration: logNow.Sub(logNow.Time)}
- backup1.Status.AvailableReplicas = &availableReplicas
- backup1.Status.Manifests = &dpv1alpha1.ManifestsStatus{
- BackupLog: &dpv1alpha1.BackupLogStatus{StartTime: &logNow, StopTime: &logNow},
- BackupTool: &dpv1alpha1.BackupToolManifestsStatus{FilePath: "/backupdata/test1"},
- Snapshot: &dpv1alpha1.BackupSnapshotStatus{VolumeSnapshotName: backupName},
- UserContext: map[string]string{"user_define_key": "user_define_value"},
- }
- backup1.Status.SourceCluster = "mycluster"
tf.FakeDynamicClient = testing.FakeDynamicClient(backup1)
Expect(o.Complete(args)).Should(Succeed())
o.client = testing.FakeClientSet()
@@ -426,13 +394,13 @@ var _ = Describe("DataProtection", func() {
})
})
-func mockBackupInfo(dynamic dynamic.Interface, backupName, clusterName string, manifests map[string]any, backupType string) {
+func mockBackupInfo(dynamic dynamic.Interface, backupName, clusterName string, timeRange map[string]any, backupMethod string) {
clusterString := fmt.Sprintf(`{"metadata":{"name":"deleted-cluster","namespace":"%s"},"spec":{"clusterDefinitionRef":"apecloud-mysql","clusterVersionRef":"ac-mysql-8.0.30","componentSpecs":[{"name":"mysql","componentDefRef":"mysql","replicas":1}]}}`, testing.Namespace)
backupStatus := &unstructured.Unstructured{
Object: map[string]any{
"status": map[string]any{
"phase": "Completed",
- "manifests": manifests,
+ "timeRange": timeRange,
},
"metadata": map[string]any{
"name": backupName,
@@ -445,7 +413,7 @@ func mockBackupInfo(dynamic dynamic.Interface, backupName, clusterName string, m
},
},
"spec": map[string]any{
- "backupType": backupType,
+ "backupMethod": backupMethod,
},
},
}
diff --git a/internal/cli/cmd/cluster/describe.go b/internal/cli/cmd/cluster/describe.go
index 8f4e1e3bd3f..c324e689fbb 100644
--- a/internal/cli/cmd/cluster/describe.go
+++ b/internal/cli/cmd/cluster/describe.go
@@ -23,11 +23,9 @@ import (
"fmt"
"io"
"strings"
- "time"
"github.com/spf13/cobra"
corev1 "k8s.io/api/core/v1"
- metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
"k8s.io/apimachinery/pkg/runtime/schema"
"k8s.io/cli-runtime/pkg/genericclioptions"
"k8s.io/client-go/dynamic"
@@ -41,7 +39,7 @@ import (
"github.com/apecloud/kubeblocks/internal/cli/printer"
"github.com/apecloud/kubeblocks/internal/cli/types"
"github.com/apecloud/kubeblocks/internal/cli/util"
- "github.com/apecloud/kubeblocks/internal/constant"
+ dptypes "github.com/apecloud/kubeblocks/internal/dataprotection/types"
)
var (
@@ -233,60 +231,60 @@ func showDataProtection(backupPolicies []dpv1alpha1.BackupPolicy, backups []dpv1
}
tbl := newTbl(out, "\nData Protection:", "AUTO-BACKUP", "BACKUP-SCHEDULE", "TYPE", "BACKUP-TTL", "LAST-SCHEDULE", "RECOVERABLE-TIME")
for _, policy := range backupPolicies {
- if policy.Annotations[constant.DefaultBackupPolicyAnnotationKey] != "true" {
+ if policy.Annotations[dptypes.DefaultBackupPolicyAnnotationKey] != "true" {
continue
}
- if policy.Status.Phase != dpv1alpha1.PolicyAvailable {
+ if policy.Status.Phase != dpv1alpha1.AvailablePhase {
continue
}
- ttlString := printer.NoneString
- backupSchedule := printer.NoneString
- backupType := printer.NoneString
- scheduleEnable := "Disabled"
- if policy.Spec.Schedule.Snapshot != nil {
- if policy.Spec.Schedule.Snapshot.Enable {
- scheduleEnable = "Enabled"
- backupSchedule = policy.Spec.Schedule.Snapshot.CronExpression
- backupType = string(dpv1alpha1.BackupTypeSnapshot)
- }
- }
- if policy.Spec.Schedule.Datafile != nil {
- if policy.Spec.Schedule.Datafile.Enable {
- scheduleEnable = "Enabled"
- backupSchedule = policy.Spec.Schedule.Datafile.CronExpression
- backupType = string(dpv1alpha1.BackupTypeDataFile)
- }
- }
- if policy.Spec.Retention != nil && policy.Spec.Retention.TTL != nil {
- ttlString = *policy.Spec.Retention.TTL
- }
- lastScheduleTime := printer.NoneString
- if policy.Status.LastScheduleTime != nil {
- lastScheduleTime = util.TimeFormat(policy.Status.LastScheduleTime)
- }
- tbl.AddRow(scheduleEnable, backupSchedule, backupType, ttlString, lastScheduleTime, getBackupRecoverableTime(backups))
+ // ttlString := printer.NoneString
+ // backupSchedule := printer.NoneString
+ // backupType := printer.NoneString
+ // scheduleEnable := "Disabled"
+ // if policy.Spec.SchedulePolicy.Snapshot != nil {
+ // if policy.Spec.SchedulePolicy.Snapshot.Enable {
+ // scheduleEnable = "Enabled"
+ // backupSchedule = policy.Spec.SchedulePolicy.Snapshot.CronExpression
+ // backupType = string(dpv1alpha1.BackupTypeSnapshot)
+ // }
+ // }
+ // if policy.Spec.SchedulePolicy.Datafile != nil {
+ // if policy.Spec.SchedulePolicy.Datafile.Enable {
+ // scheduleEnable = "Enabled"
+ // backupSchedule = policy.Spec.SchedulePolicy.Datafile.CronExpression
+ // backupType = string(dpv1alpha1.BackupTypeDataFile)
+ // }
+ // }
+ // if policy.Spec.Retention != nil && policy.Spec.Retention.TTL != nil {
+ // ttlString = *policy.Spec.Retention.TTL
+ // }
+ // lastScheduleTime := printer.NoneString
+ // if policy.Status.LastScheduleTime != nil {
+ // lastScheduleTime = util.TimeFormat(policy.Status.LastScheduleTime)
+ // }
+ // tbl.AddRow(scheduleEnable, backupSchedule, backupType, ttlString, lastScheduleTime, getBackupRecoverableTime(backups))
}
tbl.Print()
}
-// getBackupRecoverableTime returns the recoverable time range string
-func getBackupRecoverableTime(backups []dpv1alpha1.Backup) string {
- recoverabelTime := dpv1alpha1.GetRecoverableTimeRange(backups)
- var result string
- for _, i := range recoverabelTime {
- result = addTimeRange(result, i.StartTime, i.StopTime)
- }
- if result == "" {
- return printer.NoneString
- }
- return result
-}
-
-func addTimeRange(result string, start, end *metav1.Time) string {
- if result != "" {
- result += ", "
- }
- result += fmt.Sprintf("%s ~ %s", util.TimeFormatWithDuration(start, time.Second),
- util.TimeFormatWithDuration(end, time.Second))
- return result
-}
+// getBackupRecoverableTime returns the recoverable time range string
+// func getBackupRecoverableTime(backups []dpv1alpha1.Backup) string {
+// recoverabelTime := dpv1alpha1.GetRecoverableTimeRange(backups)
+// var result string
+// for _, i := range recoverabelTime {
+// result = addTimeRange(result, i.StartTime, i.StopTime)
+// }
+// if result == "" {
+// return printer.NoneString
+// }
+// return result
+// }
+
+// func addTimeRange(result string, start, end *metav1.Time) string {
+// if result != "" {
+// result += ", "
+// }
+// result += fmt.Sprintf("%s ~ %s", util.TimeFormatWithDuration(start, time.Second),
+// util.TimeFormatWithDuration(end, time.Second))
+// return result
+// }
diff --git a/internal/cli/cmd/cluster/describe_test.go b/internal/cli/cmd/cluster/describe_test.go
index 584d0b458ce..63ffe53d389 100644
--- a/internal/cli/cmd/cluster/describe_test.go
+++ b/internal/cli/cmd/cluster/describe_test.go
@@ -124,22 +124,18 @@ var _ = Describe("Expose", func() {
}
now := metav1.Now()
fakeBackups[0].Status = dpv1alpha1.BackupStatus{
- Phase: dpv1alpha1.BackupCompleted,
- Manifests: &dpv1alpha1.ManifestsStatus{
- BackupLog: &dpv1alpha1.BackupLogStatus{
- StartTime: &now,
- StopTime: &now,
- },
+ Phase: dpv1alpha1.BackupPhaseCompleted,
+ TimeRange: &dpv1alpha1.BackupTimeRange{
+ Start: &now,
+ End: &now,
},
}
after := metav1.Time{Time: now.Add(time.Hour)}
fakeBackups[1].Status = dpv1alpha1.BackupStatus{
- Phase: dpv1alpha1.BackupCompleted,
- Manifests: &dpv1alpha1.ManifestsStatus{
- BackupLog: &dpv1alpha1.BackupLogStatus{
- StartTime: &now,
- StopTime: &after,
- },
+ Phase: dpv1alpha1.BackupPhaseCompleted,
+ TimeRange: &dpv1alpha1.BackupTimeRange{
+ Start: &now,
+ End: &after,
},
}
showDataProtection(fakeBackupPolicies, fakeBackups, out)
diff --git a/internal/cli/cmd/cluster/label.go b/internal/cli/cmd/cluster/label.go
index 432536ad556..1945852b2c9 100644
--- a/internal/cli/cmd/cluster/label.go
+++ b/internal/cli/cmd/cluster/label.go
@@ -79,7 +79,6 @@ type LabelOptions struct {
namespace string
enforceNamespace bool
dryRunStrategy cmdutil.DryRunStrategy
- dryRunVerifier *resource.QueryParamVerifier
builder *resource.Builder
unstructuredClientForMapping func(mapping *meta.RESTMapping) (resource.RESTClient, error)
@@ -142,11 +141,6 @@ func (o *LabelOptions) complete(cmd *cobra.Command, args []string) error {
}
o.builder = o.Factory.NewBuilder()
o.unstructuredClientForMapping = o.Factory.UnstructuredClientForMapping
- dynamicClient, err := o.Factory.DynamicClient()
- if err != nil {
- return err
- }
- o.dryRunVerifier = resource.NewQueryParamVerifier(dynamicClient, o.Factory.OpenAPIGetter(), resource.QueryParamDryRun)
return nil
}
diff --git a/internal/cli/cmd/cluster/register.go b/internal/cli/cmd/cluster/register.go
index 3ae1bbbf69b..1bf152013fd 100644
--- a/internal/cli/cmd/cluster/register.go
+++ b/internal/cli/cmd/cluster/register.go
@@ -28,7 +28,6 @@ import (
"regexp"
"time"
- "github.com/apecloud/kubeblocks/internal/cli/util/helm"
"github.com/asaskevich/govalidator"
"github.com/spf13/cobra"
"k8s.io/cli-runtime/pkg/genericclioptions"
@@ -36,6 +35,7 @@ import (
"k8s.io/kubectl/pkg/util/templates"
"github.com/apecloud/kubeblocks/internal/cli/cluster"
+ "github.com/apecloud/kubeblocks/internal/cli/util/helm"
"github.com/apecloud/kubeblocks/internal/cli/util/prompt"
)
diff --git a/internal/cli/cmd/cluster/update.go b/internal/cli/cmd/cluster/update.go
index aeb548fc155..f8c0e10d263 100644
--- a/internal/cli/cmd/cluster/update.go
+++ b/internal/cli/cmd/cluster/update.go
@@ -43,7 +43,7 @@ import (
"k8s.io/kubectl/pkg/util/templates"
appsv1alpha1 "github.com/apecloud/kubeblocks/apis/apps/v1alpha1"
- dataprotectionv1alpha1 "github.com/apecloud/kubeblocks/apis/dataprotection/v1alpha1"
+ dpv1alpha1 "github.com/apecloud/kubeblocks/apis/dataprotection/v1alpha1"
"github.com/apecloud/kubeblocks/internal/cli/cluster"
"github.com/apecloud/kubeblocks/internal/cli/patch"
"github.com/apecloud/kubeblocks/internal/cli/types"
@@ -547,23 +547,19 @@ func (o *updateOptions) updateBackupRetentionPeriod(val string) error {
return nil
}
- // judge whether val end with the 'd'|'D'|'h'|'H' character
+ // judge whether val end with the 'd'|'h' character
lastChar := val[len(val)-1]
- if lastChar != 'd' && lastChar != 'D' && lastChar != 'h' && lastChar != 'H' {
- return fmt.Errorf("invalid retention period: %s, only support d|D|h|H", val)
+ if lastChar != 'd' && lastChar != 'h' {
+ return fmt.Errorf("invalid retention period: %s, only support d|h", val)
}
- o.cluster.Spec.Backup.RetentionPeriod = &val
+ o.cluster.Spec.Backup.RetentionPeriod = dpv1alpha1.RetentionPeriod(val)
return nil
}
func (o *updateOptions) updateBackupMethod(val string) error {
- method := dataprotectionv1alpha1.BackupMethod(val)
- if method != dataprotectionv1alpha1.BackupMethodSnapshot && method != dataprotectionv1alpha1.BackupMethodBackupTool {
- return fmt.Errorf("invalid backup method: %s, only support %s and %s", val,
- dataprotectionv1alpha1.BackupMethodSnapshot, dataprotectionv1alpha1.BackupMethodBackupTool)
- }
- o.cluster.Spec.Backup.Method = dataprotectionv1alpha1.BackupMethod(val)
+ // TODO(ldm): validate backup method are defined in the backup policy.
+ o.cluster.Spec.Backup.Method = val
return nil
}
diff --git a/internal/cli/cmd/kubeblocks/data/eks_hostpreflight.yaml b/internal/cli/cmd/kubeblocks/data/eks_hostpreflight.yaml
index ab382e8c301..09b04927af4 100644
--- a/internal/cli/cmd/kubeblocks/data/eks_hostpreflight.yaml
+++ b/internal/cli/cmd/kubeblocks/data/eks_hostpreflight.yaml
@@ -36,8 +36,12 @@ spec:
regionNames:
- cn-northwest-1
- cn-north-1
+ - us-east-1
+ - us-east-2
+ - us-west-1
+ - us-west-2
outcomes:
- warn:
- message: k8s cluster region doesn't belong to amazon china
+ message: k8s cluster region doesn't belong to amazon china, be aware of the network reachability for cluster regions that located at East-Asia
- pass:
- message: k8s cluster region belongs to amazon china
\ No newline at end of file
+ message: k8s cluster region belongs to amazon china/us
\ No newline at end of file
diff --git a/internal/cli/cmd/kubeblocks/kubeblocks_objects_test.go b/internal/cli/cmd/kubeblocks/kubeblocks_objects_test.go
index abebc2baf54..1b2ca9f6c40 100644
--- a/internal/cli/cmd/kubeblocks/kubeblocks_objects_test.go
+++ b/internal/cli/cmd/kubeblocks/kubeblocks_objects_test.go
@@ -75,35 +75,35 @@ var _ = Describe("kubeblocks objects", func() {
clusterDef.Finalizers = []string{"test"}
clusterVersion := testing.FakeClusterVersion()
clusterVersion.Finalizers = []string{"test"}
- backupTool := testing.FakeBackupTool()
- backupTool.Finalizers = []string{"test"}
+ actionSet := testing.FakeActionSet()
+ actionSet.Finalizers = []string{"test"}
testCases := []struct {
clusterDef *appsv1alpha1.ClusterDefinition
clusterVersion *appsv1alpha1.ClusterVersion
- backupTool *dpv1alpha1.BackupTool
+ actionSet *dpv1alpha1.ActionSet
}{
{
clusterDef: testing.FakeClusterDef(),
clusterVersion: testing.FakeClusterVersion(),
- backupTool: testing.FakeBackupTool(),
+ actionSet: testing.FakeActionSet(),
},
{
clusterDef: clusterDef,
clusterVersion: testing.FakeClusterVersion(),
- backupTool: testing.FakeBackupTool(),
+ actionSet: testing.FakeActionSet(),
},
{
clusterDef: clusterDef,
clusterVersion: clusterVersion,
- backupTool: backupTool,
+ actionSet: actionSet,
},
}
for _, c := range testCases {
objects := mockCRD()
objects = append(objects, testing.FakeVolumeSnapshotClass())
- objects = append(objects, c.clusterDef, c.clusterVersion, c.backupTool)
+ objects = append(objects, c.clusterDef, c.clusterVersion, c.actionSet)
client := testing.FakeDynamicClient(objects...)
objs, _ := getKBObjects(client, "", nil)
Expect(removeCustomResources(client, objs)).Should(Succeed())
@@ -135,22 +135,22 @@ var _ = Describe("kubeblocks objects", func() {
Expect(objs[types.CRDGVR()].Items).Should(HaveLen(4))
// verify crs
for _, gvr := range []schema.GroupVersionResource{types.ClusterDefGVR(), types.ClusterVersionGVR()} {
- objlist, ok := objs[gvr]
+ objList, ok := objs[gvr]
Expect(ok).Should(BeTrue())
- Expect(objlist.Items).Should(HaveLen(1))
+ Expect(objList.Items).Should(HaveLen(1))
}
// verify rbac info
for _, gvr := range []schema.GroupVersionResource{types.RoleGVR(), types.ClusterRoleBindingGVR(), types.ServiceAccountGVR()} {
- objlist, ok := objs[gvr]
+ objList, ok := objs[gvr]
Expect(ok).Should(BeTrue())
- Expect(objlist.Items).Should(HaveLen(1), gvr.String())
+ Expect(objList.Items).Should(HaveLen(1), gvr.String())
}
- // verify cofnig tpl
+ // verify config tpl
for _, gvr := range []schema.GroupVersionResource{types.ConfigmapGVR()} {
- objlist, ok := objs[gvr]
+ objList, ok := objs[gvr]
Expect(ok).Should(BeTrue())
- Expect(objlist.Items).Should(HaveLen(1), gvr.String())
+ Expect(objList.Items).Should(HaveLen(1), gvr.String())
}
})
})
@@ -200,20 +200,20 @@ func mockCRD() []runtime.Object {
Status: v1.CustomResourceDefinitionStatus{},
}
- backupToolCRD := v1.CustomResourceDefinition{
+ actionSetCRD := v1.CustomResourceDefinition{
TypeMeta: metav1.TypeMeta{
Kind: "CustomResourceDefinition",
APIVersion: "apiextensions.k8s.io/v1",
},
ObjectMeta: metav1.ObjectMeta{
- Name: "backuptools.dataprotection.kubeblocks.io",
+ Name: "actionsets.dataprotection.kubeblocks.io",
},
Spec: v1.CustomResourceDefinitionSpec{
Group: types.DPAPIGroup,
},
Status: v1.CustomResourceDefinitionStatus{},
}
- return []runtime.Object{&clusterCRD, &clusterDefCRD, &clusterVersionCRD, &backupToolCRD}
+ return []runtime.Object{&clusterCRD, &clusterDefCRD, &clusterVersionCRD, &actionSetCRD}
}
func mockCRs() []runtime.Object {
diff --git a/internal/cli/cmd/kubeblocks/status.go b/internal/cli/cmd/kubeblocks/status.go
index 33da9bc955e..526d37eeac8 100644
--- a/internal/cli/cmd/kubeblocks/status.go
+++ b/internal/cli/cmd/kubeblocks/status.go
@@ -70,7 +70,7 @@ var (
}
kubeBlocksGlobalCustomResources = []schema.GroupVersionResource{
- types.BackupToolGVR(),
+ types.ActionSetGVR(),
types.ClusterDefGVR(),
types.ClusterVersionGVR(),
types.ConfigConstraintGVR(),
diff --git a/internal/cli/cmd/kubeblocks/upgrade.go b/internal/cli/cmd/kubeblocks/upgrade.go
index 3592ae09457..667ce1d5c13 100644
--- a/internal/cli/cmd/kubeblocks/upgrade.go
+++ b/internal/cli/cmd/kubeblocks/upgrade.go
@@ -120,7 +120,8 @@ func (o *InstallOptions) Upgrade() error {
}
// double check for KubeBlocks upgrade
- if !o.autoApprove {
+ // and only check when KubeBlocks version change
+ if !o.autoApprove && o.Version != "" {
oldVersion, err := version.NewVersion(kbVersion)
if err != nil {
return err
diff --git a/internal/cli/cmd/kubeblocks/upgrade_test.go b/internal/cli/cmd/kubeblocks/upgrade_test.go
index e3bbeb2d0d0..4dabe696c8a 100644
--- a/internal/cli/cmd/kubeblocks/upgrade_test.go
+++ b/internal/cli/cmd/kubeblocks/upgrade_test.go
@@ -72,7 +72,7 @@ var _ = Describe("kubeblocks upgrade", func() {
Expect(o.Namespace).To(Equal("test"))
})
- It("run upgrade double-check", func() {
+ It("double-check when version change", func() {
mockDeploy := func() *appsv1.Deployment {
deploy := &appsv1.Deployment{}
deploy.SetLabels(map[string]string{
@@ -101,6 +101,30 @@ var _ = Describe("kubeblocks upgrade", func() {
})
+ It("helm ValueOpts upgrade", func() {
+ mockDeploy := func() *appsv1.Deployment {
+ deploy := &appsv1.Deployment{}
+ deploy.SetLabels(map[string]string{
+ "app.kubernetes.io/name": types.KubeBlocksChartName,
+ "app.kubernetes.io/version": "0.3.0",
+ })
+ return deploy
+ }
+
+ o := &InstallOptions{
+ Options: Options{
+ IOStreams: streams,
+ HelmCfg: helm.NewFakeConfig(namespace),
+ Namespace: "default",
+ Client: testing.FakeClientSet(mockDeploy()),
+ Dynamic: testing.FakeDynamicClient(),
+ },
+ Version: "",
+ }
+ o.ValueOpts.Values = []string{"replicaCount=2"}
+ Expect(o.Upgrade()).Should(Succeed())
+ })
+
It("run upgrade", func() {
mockDeploy := func() *appsv1.Deployment {
deploy := &appsv1.Deployment{}
diff --git a/internal/cli/cmd/plugin/types.go b/internal/cli/cmd/plugin/types.go
index e05767dc8e4..1480511352c 100644
--- a/internal/cli/cmd/plugin/types.go
+++ b/internal/cli/cmd/plugin/types.go
@@ -67,6 +67,9 @@ func (p *Paths) IndexPluginsPath(name string) []string {
if _, err := os.Stat(filepath.Join(p.IndexPath(name), "krew-plugins")); err == nil {
result = append(result, filepath.Join(p.IndexPath(name), "krew-plugins"))
}
+ if _, err := os.Stat(filepath.Join(p.IndexPath(name), "cli-plugins")); err == nil {
+ result = append(result, filepath.Join(p.IndexPath(name), "cli-plugins"))
+ }
return result
}
diff --git a/internal/cli/cmd/report/report.go b/internal/cli/cmd/report/report.go
index 803dab4118a..48c20c18eae 100644
--- a/internal/cli/cmd/report/report.go
+++ b/internal/cli/cmd/report/report.go
@@ -33,7 +33,7 @@ import (
utilerrors "k8s.io/apimachinery/pkg/util/errors"
"k8s.io/cli-runtime/pkg/genericclioptions"
"k8s.io/cli-runtime/pkg/printers"
- klog "k8s.io/klog/v2"
+ "k8s.io/klog/v2"
cmdutil "k8s.io/kubectl/pkg/cmd/util"
"k8s.io/kubectl/pkg/util"
"k8s.io/kubectl/pkg/util/i18n"
@@ -348,6 +348,8 @@ func (o *reportKubeblocksOptions) handleManifests(ctx context.Context) error {
resourceLists = append(resourceLists, cliutil.ListResourceByGVR(ctx, o.genericClientSet.dynamic, o.namespace, scopedgvrs, []metav1.ListOptions{o.kubeBlocksSelector}, &allErrors)...)
// get global resources
resourceLists = append(resourceLists, cliutil.ListResourceByGVR(ctx, o.genericClientSet.dynamic, metav1.NamespaceAll, globalGvrs, []metav1.ListOptions{o.kubeBlocksSelector}, &allErrors)...)
+ // get all storage class
+ resourceLists = append(resourceLists, cliutil.ListResourceByGVR(ctx, o.genericClientSet.dynamic, metav1.NamespaceAll, []schema.GroupVersionResource{types.StorageClassGVR()}, []metav1.ListOptions{{}}, &allErrors)...)
if err := o.reportWritter.WriteObjects(manifestsFolder, resourceLists, o.outputFormat); err != nil {
return err
}
@@ -490,10 +492,16 @@ func (o *reportClusterOptions) handleManifests(ctx context.Context) error {
types.RoleBindingGVR(),
types.BackupGVR(),
types.BackupPolicyGVR(),
- types.BackupToolGVR(),
- types.RestoreJobGVR(),
+ types.BackupScheduleGVR(),
+ types.ActionSetGVR(),
+ types.RestoreGVR(),
+ types.PVCGVR(),
+ }
+ globalGvrs = []schema.GroupVersionResource{
+ types.PVGVR(),
}
)
+
var err error
if o.cluster, err = o.genericClientSet.kbClientSet.AppsV1alpha1().Clusters(o.namespace).Get(ctx, o.clusterName, metav1.GetOptions{}); err != nil {
return err
@@ -508,6 +516,7 @@ func (o *reportClusterOptions) handleManifests(ctx context.Context) error {
resourceLists := make([]*unstructured.UnstructuredList, 0)
// write manifest
resourceLists = append(resourceLists, cliutil.ListResourceByGVR(ctx, o.genericClientSet.dynamic, o.namespace, scopedgvrs, []metav1.ListOptions{o.clusterSelector}, &allErrors)...)
+ resourceLists = append(resourceLists, cliutil.ListResourceByGVR(ctx, o.genericClientSet.dynamic, metav1.NamespaceAll, globalGvrs, []metav1.ListOptions{o.clusterSelector}, &allErrors)...)
if err := o.reportWritter.WriteObjects("manifests", resourceLists, o.outputFormat); err != nil {
return err
}
diff --git a/internal/cli/create/template/backup_template.cue b/internal/cli/create/template/backup_template.cue
index 85d37eabf71..5d0b80f9d56 100644
--- a/internal/cli/create/template/backup_template.cue
+++ b/internal/cli/create/template/backup_template.cue
@@ -19,7 +19,7 @@
options: {
backupName: string
namespace: string
- backupType: string
+ backupMethod: string
backupPolicy: string
}
@@ -35,7 +35,7 @@ content: {
}
}
spec: {
- backupType: options.backupType
+ backupMethod: options.backupMethod
backupPolicyName: options.backupPolicy
}
}
diff --git a/internal/cli/edit/custom_edit.go b/internal/cli/edit/custom_edit.go
index 7f15a05046c..3c16f064d5a 100644
--- a/internal/cli/edit/custom_edit.go
+++ b/internal/cli/edit/custom_edit.go
@@ -32,7 +32,6 @@ import (
"k8s.io/apimachinery/pkg/runtime"
"k8s.io/apimachinery/pkg/util/yaml"
"k8s.io/cli-runtime/pkg/genericclioptions"
- "k8s.io/cli-runtime/pkg/resource"
cmdutil "k8s.io/kubectl/pkg/cmd/util"
"k8s.io/kubectl/pkg/cmd/util/editor"
@@ -88,13 +87,8 @@ func (o *CustomEditOptions) Run(originalObj runtime.Object) error {
edited = original
}
- dynamicClient, err := o.Factory.DynamicClient()
- if err != nil {
- return fmt.Errorf("failed to get dynamic client: %v", err)
- }
// apply validation
- fieldValidationVerifier := resource.NewQueryParamVerifier(dynamicClient, o.Factory.OpenAPIGetter(), resource.QueryParamFieldValidation)
- schemaValidator, err := o.Factory.Validator(metav1.FieldValidationStrict, fieldValidationVerifier)
+ schemaValidator, err := o.Factory.Validator(metav1.FieldValidationStrict)
if err != nil {
return fmt.Errorf("failed to get validator: %v", err)
}
diff --git a/internal/cli/patch/patch.go b/internal/cli/patch/patch.go
index eeadffcedf1..ae3a90cd96e 100644
--- a/internal/cli/patch/patch.go
+++ b/internal/cli/patch/patch.go
@@ -65,7 +65,6 @@ type Options struct {
namespace string
enforceNamespace bool
dryRunStrategy cmdutil.DryRunStrategy
- dryRunVerifier *resource.QueryParamVerifier
args []string
builder *resource.Builder
unstructuredClientForMapping func(mapping *meta.RESTMapping) (resource.RESTClient, error)
@@ -116,11 +115,6 @@ func (o *Options) complete(cmd *cobra.Command) error {
o.args = append([]string{util.GVRToString(o.GVR)}, o.Names...)
o.builder = o.Factory.NewBuilder()
o.unstructuredClientForMapping = o.Factory.UnstructuredClientForMapping
- dynamicClient, err := o.Factory.DynamicClient()
- if err != nil {
- return err
- }
- o.dryRunVerifier = resource.NewQueryParamVerifier(dynamicClient, o.Factory.OpenAPIGetter(), resource.QueryParamDryRun)
return nil
}
@@ -161,11 +155,6 @@ func (o *Options) Run(cmd *cobra.Command) error {
name, namespace := info.Name, info.Namespace
if o.dryRunStrategy != cmdutil.DryRunClient {
mapping := info.ResourceMapping()
- if o.dryRunStrategy == cmdutil.DryRunServer {
- if err := o.dryRunVerifier.HasSupport(mapping.GroupVersionKind); err != nil {
- return err
- }
- }
client, err := o.unstructuredClientForMapping(mapping)
if err != nil {
return err
diff --git a/internal/cli/scheme/install.go b/internal/cli/scheme/install.go
index 7105830579b..185e51edf22 100644
--- a/internal/cli/scheme/install.go
+++ b/internal/cli/scheme/install.go
@@ -28,7 +28,7 @@ import (
clientgoscheme "k8s.io/client-go/kubernetes/scheme"
appsv1alpha1 "github.com/apecloud/kubeblocks/apis/apps/v1alpha1"
- dataprotectionv1alpha1 "github.com/apecloud/kubeblocks/apis/dataprotection/v1alpha1"
+ dpv1alpha1 "github.com/apecloud/kubeblocks/apis/dataprotection/v1alpha1"
extensionsv1alpha1 "github.com/apecloud/kubeblocks/apis/extensions/v1alpha1"
workloadsv1alpha1 "github.com/apecloud/kubeblocks/apis/workloads/v1alpha1"
)
@@ -37,7 +37,7 @@ func init() {
utilruntime.Must(metav1.AddMetaToScheme(Scheme))
utilruntime.Must(clientgoscheme.AddToScheme(Scheme))
utilruntime.Must(appsv1alpha1.AddToScheme(Scheme))
- utilruntime.Must(dataprotectionv1alpha1.AddToScheme(Scheme))
+ utilruntime.Must(dpv1alpha1.AddToScheme(Scheme))
utilruntime.Must(snapshotv1.AddToScheme(Scheme))
utilruntime.Must(snapshotv1beta1.AddToScheme(Scheme))
utilruntime.Must(extensionsv1alpha1.AddToScheme(Scheme))
diff --git a/internal/cli/testing/fake.go b/internal/cli/testing/fake.go
index 8e98600f3d7..f300935e8a0 100644
--- a/internal/cli/testing/fake.go
+++ b/internal/cli/testing/fake.go
@@ -44,6 +44,8 @@ import (
storagev1alpha1 "github.com/apecloud/kubeblocks/apis/storage/v1alpha1"
"github.com/apecloud/kubeblocks/internal/cli/types"
"github.com/apecloud/kubeblocks/internal/constant"
+ dptypes "github.com/apecloud/kubeblocks/internal/dataprotection/types"
+ "github.com/apecloud/kubeblocks/internal/dataprotection/utils/boolptr"
testapps "github.com/apecloud/kubeblocks/internal/testutil/apps"
)
@@ -62,7 +64,9 @@ const (
KubeBlocksRepoName = "fake-kubeblocks-repo"
KubeBlocksChartName = "fake-kubeblocks"
KubeBlocksChartURL = "fake-kubeblocks-chart-url"
- BackupToolName = "fake-backup-tool"
+ BackupMethodName = "fake-backup-method"
+ ActionSetName = "fake-action-set"
+ BackupName = "fake-backup-name"
IsDefault = true
IsNotDefault = false
@@ -356,14 +360,13 @@ func FakeClusterVersion() *appsv1alpha1.ClusterVersion {
return cv
}
-func FakeBackupTool() *dpv1alpha1.BackupTool {
- tool := &dpv1alpha1.BackupTool{}
- tool.Name = BackupToolName
- return tool
+func FakeActionSet() *dpv1alpha1.ActionSet {
+ as := &dpv1alpha1.ActionSet{}
+ as.Name = ActionSetName
+ return as
}
func FakeBackupPolicy(backupPolicyName, clusterName string) *dpv1alpha1.BackupPolicy {
- ttl := "7d"
template := &dpv1alpha1.BackupPolicy{
TypeMeta: metav1.TypeMeta{
APIVersion: fmt.Sprintf("%s/%s", types.DPAPIGroup, types.DPAPIVersion),
@@ -376,51 +379,30 @@ func FakeBackupPolicy(backupPolicyName, clusterName string) *dpv1alpha1.BackupPo
constant.AppInstanceLabelKey: clusterName,
},
Annotations: map[string]string{
- constant.DefaultBackupPolicyAnnotationKey: "true",
+ dptypes.DefaultBackupPolicyAnnotationKey: "true",
},
},
Spec: dpv1alpha1.BackupPolicySpec{
- Snapshot: &dpv1alpha1.SnapshotPolicy{
- BasePolicy: dpv1alpha1.BasePolicy{
- BackupsHistoryLimit: 1,
- },
- },
- Datafile: &dpv1alpha1.CommonBackupPolicy{
- BasePolicy: dpv1alpha1.BasePolicy{
- BackupsHistoryLimit: 1,
- },
- PersistentVolumeClaim: dpv1alpha1.PersistentVolumeClaim{
- Name: pointer.String("test1"),
- },
- },
- Logfile: &dpv1alpha1.CommonBackupPolicy{
- BasePolicy: dpv1alpha1.BasePolicy{
- BackupsHistoryLimit: 1,
- },
- PersistentVolumeClaim: dpv1alpha1.PersistentVolumeClaim{
- Name: pointer.String("test1"),
+ BackupMethods: []dpv1alpha1.BackupMethod{
+ {
+ Name: BackupMethodName,
+ SnapshotVolumes: boolptr.False(),
+ ActionSetName: ActionSetName,
},
},
- Schedule: dpv1alpha1.Schedule{
- Snapshot: &dpv1alpha1.SchedulePolicy{
- Enable: false,
- CronExpression: "0 18 * * *",
- },
- Datafile: &dpv1alpha1.SchedulePolicy{
- Enable: false,
- CronExpression: "0 18 * * *",
- },
- Logfile: &dpv1alpha1.SchedulePolicy{
- Enable: false,
- CronExpression: "* */1 * * *",
+ Target: &dpv1alpha1.BackupTarget{
+ PodSelector: &dpv1alpha1.PodSelector{
+ LabelSelector: &metav1.LabelSelector{
+ MatchLabels: map[string]string{
+ constant.AppInstanceLabelKey: ClusterName,
+ constant.KBAppComponentLabelKey: ComponentName,
+ constant.AppManagedByLabelKey: constant.AppName},
+ },
},
},
- Retention: &dpv1alpha1.RetentionSpec{
- TTL: &ttl,
- },
},
Status: dpv1alpha1.BackupPolicyStatus{
- Phase: dpv1alpha1.PolicyAvailable,
+ Phase: dpv1alpha1.AvailablePhase,
},
}
return template
@@ -451,8 +433,8 @@ func FakeBackupWithCluster(cluster *appsv1alpha1.Cluster, backupName string) *dp
Name: backupName,
Namespace: Namespace,
Labels: map[string]string{
- constant.AppInstanceLabelKey: cluster.Name,
- constant.DataProtectionLabelClusterUIDKey: string(cluster.UID),
+ constant.AppInstanceLabelKey: cluster.Name,
+ dptypes.DataProtectionLabelClusterUIDKey: string(cluster.UID),
},
},
}
@@ -1025,7 +1007,7 @@ func FakeBackupRepo(name string, isDefault bool) *dpv1alpha1.BackupRepo {
}
if isDefault {
backupRepo.Annotations = map[string]string{
- constant.DefaultBackupRepoAnnotationKey: "true",
+ dptypes.DefaultBackupRepoAnnotationKey: "true",
}
}
return backupRepo
diff --git a/internal/cli/types/types.go b/internal/cli/types/types.go
index 328b1b91a0d..d392d9702b9 100644
--- a/internal/cli/types/types.go
+++ b/internal/cli/types/types.go
@@ -103,7 +103,7 @@ const (
KindClusterVersion = "ClusterVersion"
KindConfigConstraint = "ConfigConstraint"
KindBackup = "Backup"
- KindRestoreJob = "RestoreJob"
+ KindRestore = "Restore"
KindBackupPolicy = "BackupPolicy"
KindOps = "OpsRequest"
)
@@ -132,13 +132,14 @@ const (
// DataProtection API group
const (
- DPAPIGroup = "dataprotection.kubeblocks.io"
- DPAPIVersion = "v1alpha1"
- ResourceBackups = "backups"
- ResourceBackupTools = "backuptools"
- ResourceRestoreJobs = "restorejobs"
- ResourceBackupPolicies = "backuppolicies"
- ResourceBackupRepos = "backuprepos"
+ DPAPIGroup = "dataprotection.kubeblocks.io"
+ DPAPIVersion = "v1alpha1"
+ ResourceBackups = "backups"
+ ResourceActionSets = "actionsets"
+ ResourceRestores = "restores"
+ ResourceBackupPolicies = "backuppolicies"
+ ResourceBackupRepos = "backuprepos"
+ ResourceBackupSchedules = "backupschedules"
)
// Extensions API group
@@ -251,16 +252,20 @@ func BackupPolicyGVR() schema.GroupVersionResource {
return schema.GroupVersionResource{Group: DPAPIGroup, Version: DPAPIVersion, Resource: ResourceBackupPolicies}
}
-func BackupToolGVR() schema.GroupVersionResource {
- return schema.GroupVersionResource{Group: DPAPIGroup, Version: DPAPIVersion, Resource: ResourceBackupTools}
+func BackupScheduleGVR() schema.GroupVersionResource {
+ return schema.GroupVersionResource{Group: DPAPIGroup, Version: DPAPIVersion, Resource: ResourceBackupSchedules}
+}
+
+func ActionSetGVR() schema.GroupVersionResource {
+ return schema.GroupVersionResource{Group: DPAPIGroup, Version: DPAPIVersion, Resource: ResourceActionSets}
}
func BackupRepoGVR() schema.GroupVersionResource {
return schema.GroupVersionResource{Group: DPAPIGroup, Version: DPAPIVersion, Resource: ResourceBackupRepos}
}
-func RestoreJobGVR() schema.GroupVersionResource {
- return schema.GroupVersionResource{Group: DPAPIGroup, Version: DPAPIVersion, Resource: ResourceRestoreJobs}
+func RestoreGVR() schema.GroupVersionResource {
+ return schema.GroupVersionResource{Group: DPAPIGroup, Version: DPAPIVersion, Resource: ResourceRestores}
}
func AddonGVR() schema.GroupVersionResource {
diff --git a/internal/cli/util/completion.go b/internal/cli/util/completion.go
index b29f692b916..6e6e61d17eb 100644
--- a/internal/cli/util/completion.go
+++ b/internal/cli/util/completion.go
@@ -39,7 +39,7 @@ import (
func ResourceNameCompletionFunc(f cmdutil.Factory, gvr schema.GroupVersionResource) func(*cobra.Command, []string, string) ([]string, cobra.ShellCompDirective) {
return func(cmd *cobra.Command, args []string, toComplete string) ([]string, cobra.ShellCompDirective) {
- comps := utilcomp.CompGetResource(f, cmd, GVRToString(gvr), toComplete)
+ comps := utilcomp.CompGetResource(f, GVRToString(gvr), toComplete)
seen := make(map[string]bool)
var availableComps []string
@@ -103,7 +103,7 @@ func CompGetFromTemplateWithLabels(template *string, f cmdutil.Factory, namespac
o.LabelSelector = strings.Join(labels, ",")
}
- _ = o.Run(f, cmd, args)
+ _ = o.Run(f, args)
var comps []string
resources := strings.Split(buf.String(), " ")
@@ -123,7 +123,7 @@ func RegisterClusterCompletionFunc(cmd *cobra.Command, f cmdutil.Factory) {
cmdutil.CheckErr(cmd.RegisterFlagCompletionFunc(
"cluster",
func(cmd *cobra.Command, args []string, toComplete string) ([]string, cobra.ShellCompDirective) {
- return utilcomp.CompGetResource(f, cmd, GVRToString(types.ClusterGVR()), toComplete), cobra.ShellCompDirectiveNoFileComp
+ return utilcomp.CompGetResource(f, GVRToString(types.ClusterGVR()), toComplete), cobra.ShellCompDirectiveNoFileComp
},
))
}
diff --git a/internal/cli/util/flags/flags.go b/internal/cli/util/flags/flags.go
index 5335877cf07..3d4c9000662 100644
--- a/internal/cli/util/flags/flags.go
+++ b/internal/cli/util/flags/flags.go
@@ -40,7 +40,7 @@ func AddClusterDefinitionFlag(f cmdutil.Factory, cmd *cobra.Command, p *string)
cmd.Flags().StringVar(p, "cluster-definition", *p, "Specify cluster definition, run \"kbcli clusterdefinition list\" to show all available cluster definition")
util.CheckErr(cmd.RegisterFlagCompletionFunc("cluster-definition",
func(cmd *cobra.Command, args []string, toComplete string) ([]string, cobra.ShellCompDirective) {
- return utilcomp.CompGetResource(f, cmd, util.GVRToString(types.ClusterDefGVR()), toComplete), cobra.ShellCompDirectiveNoFileComp
+ return utilcomp.CompGetResource(f, util.GVRToString(types.ClusterDefGVR()), toComplete), cobra.ShellCompDirectiveNoFileComp
}))
}
diff --git a/internal/cli/util/helm/helm.go b/internal/cli/util/helm/helm.go
index a2ee2dde334..d0fead1c0de 100644
--- a/internal/cli/util/helm/helm.go
+++ b/internal/cli/util/helm/helm.go
@@ -182,7 +182,7 @@ func (i *InstallOpts) Install(cfg *Config) (*release.Release, error) {
}
var rel *release.Release
- if err := retry.IfNecessary(ctx, func() error {
+ if err = retry.IfNecessary(ctx, func() error {
release, err1 := i.tryInstall(actionCfg)
if err1 != nil {
return err1
@@ -212,7 +212,8 @@ func (i *InstallOpts) tryInstall(cfg *action.Configuration) (*release.Release, e
// If a release does not exist, install it.
histClient := action.NewHistory(cfg)
histClient.Max = 1
- if _, err := histClient.Run(i.Name); err != nil && err != driver.ErrReleaseNotFound {
+ if _, err := histClient.Run(i.Name); err != nil &&
+ !errors.Is(err, driver.ErrReleaseNotFound) {
return nil, err
}
diff --git a/internal/cli/util/version.go b/internal/cli/util/version.go
index 9f2cd2ed4d8..ae24154ae4a 100644
--- a/internal/cli/util/version.go
+++ b/internal/cli/util/version.go
@@ -20,6 +20,7 @@ along with this program. If not, see .
package util
import (
+ "bytes"
"context"
"fmt"
"os/exec"
@@ -128,9 +129,15 @@ func GetKubeBlocksDeploy(client kubernetes.Interface) (*appsv1.Deployment, error
func GetDockerVersion() (*gv.Version, error) {
// exec cmd to get output from docker info --format '{{.ServerVersion}}'
cmd := exec.Command("docker", "info", "--format", "{{.ServerVersion}}")
+ var stderr bytes.Buffer
+ cmd.Stderr = &stderr
out, err := cmd.Output()
- if err != nil {
- return nil, err
+ if err != nil || stderr.String() != "" {
+ errMsg := stderr.String()
+ if errMsg == "" {
+ errMsg = err.Error()
+ }
+ return nil, fmt.Errorf("failed to get the docker version by executing \"docker info --format {{.ServerVersion}}\": %s", errMsg)
}
return gv.NewVersion(strings.TrimSpace(string(out)))
}
diff --git a/internal/common/doc.go b/internal/common/doc.go
new file mode 100644
index 00000000000..878d6bf35ab
--- /dev/null
+++ b/internal/common/doc.go
@@ -0,0 +1,24 @@
+/*
+Copyright (C) 2022-2023 ApeCloud Co., Ltd
+
+This file is part of KubeBlocks project
+
+This program is free software: you can redistribute it and/or modify
+it under the terms of the GNU Affero General Public License as published by
+the Free Software Foundation, either version 3 of the License, or
+(at your option) any later version.
+
+This program is distributed in the hope that it will be useful
+but WITHOUT ANY WARRANTY; without even the implied warranty of
+MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+GNU Affero General Public License for more details.
+
+You should have received a copy of the GNU Affero General Public License
+along with this program. If not, see .
+*/
+
+/*
+Package common provides types and utils shared by all KubeBlocks components: KubeBlocks Core, KBCLI, Lorry etc.
+will promote to pkg/common when stable.
+*/
+package common
diff --git a/controllers/apps/components/stateful_set_utils.go b/internal/common/stateful_set_utils.go
similarity index 57%
rename from controllers/apps/components/stateful_set_utils.go
rename to internal/common/stateful_set_utils.go
index 0d7f7630164..da13b3dcb76 100644
--- a/controllers/apps/components/stateful_set_utils.go
+++ b/internal/common/stateful_set_utils.go
@@ -17,7 +17,7 @@ You should have received a copy of the GNU Affero General Public License
along with this program. If not, see .
*/
-package components
+package common
import (
"context"
@@ -26,10 +26,9 @@ import (
appsv1 "k8s.io/api/apps/v1"
corev1 "k8s.io/api/core/v1"
- apierrors "k8s.io/apimachinery/pkg/api/errors"
+ metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
"sigs.k8s.io/controller-runtime/pkg/client"
- "github.com/apecloud/kubeblocks/internal/constant"
intctrlutil "github.com/apecloud/kubeblocks/internal/controllerutil"
)
@@ -50,69 +49,6 @@ func IsMemberOf(set *appsv1.StatefulSet, pod *corev1.Pod) bool {
return getParentName(pod) == set.Name
}
-// isStsAndPodsRevisionConsistent checks if StatefulSet and pods of the StatefulSet have the same revision.
-func isStsAndPodsRevisionConsistent(ctx context.Context, cli client.Client, sts *appsv1.StatefulSet) (bool, error) {
- pods, err := GetPodListByStatefulSet(ctx, cli, sts)
- if err != nil {
- return false, err
- }
-
- revisionConsistent := true
- if len(pods) != int(*sts.Spec.Replicas) {
- return false, nil
- }
-
- for _, pod := range pods {
- if intctrlutil.GetPodRevision(&pod) != sts.Status.UpdateRevision {
- revisionConsistent = false
- break
- }
- }
- return revisionConsistent, nil
-}
-
-// getPods4Delete gets all pods for delete
-func getPods4Delete(ctx context.Context, cli client.Client, sts *appsv1.StatefulSet) ([]*corev1.Pod, error) {
- if sts.Spec.UpdateStrategy.Type == appsv1.RollingUpdateStatefulSetStrategyType {
- return nil, nil
- }
-
- pods, err := GetPodListByStatefulSet(ctx, cli, sts)
- if err != nil {
- return nil, nil
- }
-
- podList := make([]*corev1.Pod, 0)
- for i, pod := range pods {
- // do nothing if the pod is terminating
- if pod.DeletionTimestamp != nil {
- continue
- }
- // do nothing if the pod has the latest version
- if intctrlutil.GetPodRevision(&pod) == sts.Status.UpdateRevision {
- continue
- }
-
- podList = append(podList, &pods[i])
- }
- return podList, nil
-}
-
-// deleteStsPods deletes pods of the StatefulSet manually
-func deleteStsPods(ctx context.Context, cli client.Client, sts *appsv1.StatefulSet) error {
- pods, err := getPods4Delete(ctx, cli, sts)
- if err != nil {
- return err
- }
- for _, pod := range pods {
- // delete the pod to trigger associate StatefulSet to re-create it
- if err := cli.Delete(ctx, pod); err != nil && !apierrors.IsNotFound(err) {
- return err
- }
- }
- return nil
-}
-
// statefulSetOfComponentIsReady checks if statefulSet of component is ready.
func statefulSetOfComponentIsReady(sts *appsv1.StatefulSet, statefulStatusRevisionIsEquals bool, targetReplicas *int32) bool {
if targetReplicas == nil {
@@ -157,13 +93,19 @@ func ParseParentNameAndOrdinal(s string) (string, int32) {
// GetPodListByStatefulSet gets statefulSet pod list.
func GetPodListByStatefulSet(ctx context.Context, cli client.Client, stsObj *appsv1.StatefulSet) ([]corev1.Pod, error) {
+ selector, err := metav1.LabelSelectorAsMap(stsObj.Spec.Selector)
+ if err != nil {
+ return nil, err
+ }
+ return GetPodListByStatefulSetWithSelector(ctx, cli, stsObj, selector)
+}
+
+// GetPodListByStatefulSetWithSelector gets statefulSet pod list.
+func GetPodListByStatefulSetWithSelector(ctx context.Context, cli client.Client, stsObj *appsv1.StatefulSet, selector client.MatchingLabels) ([]corev1.Pod, error) {
podList := &corev1.PodList{}
if err := cli.List(ctx, podList,
&client.ListOptions{Namespace: stsObj.Namespace},
- client.MatchingLabels{
- constant.KBAppComponentLabelKey: stsObj.Labels[constant.KBAppComponentLabelKey],
- constant.AppInstanceLabelKey: stsObj.Labels[constant.AppInstanceLabelKey],
- }); err != nil {
+ selector); err != nil {
return nil, err
}
var pods []corev1.Pod
@@ -174,22 +116,3 @@ func GetPodListByStatefulSet(ctx context.Context, cli client.Client, stsObj *app
}
return pods, nil
}
-
-// getPodOwnerReferencesSts gets the owner reference statefulSet of the pod.
-func getPodOwnerReferencesSts(ctx context.Context, cli client.Client, podObj *corev1.Pod) (*appsv1.StatefulSet, error) {
- stsList := &appsv1.StatefulSetList{}
- if err := cli.List(ctx, stsList,
- &client.ListOptions{Namespace: podObj.Namespace},
- client.MatchingLabels{
- constant.KBAppComponentLabelKey: podObj.Labels[constant.KBAppComponentLabelKey],
- constant.AppInstanceLabelKey: podObj.Labels[constant.AppInstanceLabelKey],
- }); err != nil {
- return nil, err
- }
- for _, sts := range stsList.Items {
- if IsMemberOf(&sts, podObj) {
- return &sts, nil
- }
- }
- return nil, nil
-}
diff --git a/controllers/apps/components/stateful_set_utils_test.go b/internal/common/stateful_set_utils_test.go
similarity index 50%
rename from controllers/apps/components/stateful_set_utils_test.go
rename to internal/common/stateful_set_utils_test.go
index 25b1efaccdd..3c8c0e925cf 100644
--- a/controllers/apps/components/stateful_set_utils_test.go
+++ b/internal/common/stateful_set_utils_test.go
@@ -17,21 +17,14 @@ You should have received a copy of the GNU Affero General Public License
along with this program. If not, see .
*/
-package components
+package common
import (
"testing"
- . "github.com/onsi/ginkgo/v2"
- . "github.com/onsi/gomega"
-
apps "k8s.io/api/apps/v1"
- corev1 "k8s.io/api/core/v1"
- "sigs.k8s.io/controller-runtime/pkg/client"
intctrlutil "github.com/apecloud/kubeblocks/internal/controllerutil"
- "github.com/apecloud/kubeblocks/internal/generics"
- testapps "github.com/apecloud/kubeblocks/internal/testutil/apps"
testk8s "github.com/apecloud/kubeblocks/internal/testutil/k8s"
)
@@ -97,65 +90,3 @@ func TestSStatefulSetOfComponentIsReady(t *testing.T) {
t.Errorf("StatefulSet should not be ready")
}
}
-
-var _ = Describe("StatefulSet utils test", func() {
- var (
- clusterName = "test-replication-cluster"
- stsName = "test-sts"
- role = "Primary"
- )
- cleanAll := func() {
- By("Cleaning resources")
- // delete cluster(and all dependent sub-resources), clusterversion and clusterdef
- testapps.ClearClusterResources(&testCtx)
- // clear rest resources
- inNS := client.InNamespace(testCtx.DefaultNamespace)
- ml := client.HasLabels{testCtx.TestObjLabelKey}
- // namespaced resources
- // testapps.ClearResources(&testCtx, generics.StatefulSetSignature, inNS, ml)
- testapps.ClearResources(&testCtx, generics.PodSignature, inNS, ml, client.GracePeriodSeconds(0))
- }
-
- BeforeEach(cleanAll)
- AfterEach(cleanAll)
-
- When("Updating a StatefulSet with `OnDelete` UpdateStrategy", func() {
- It("will not update pods of the StatefulSet util the pods have been manually deleted", func() {
- By("Creating a StatefulSet")
- sts := testapps.NewStatefulSetFactory(testCtx.DefaultNamespace, stsName, clusterName, testapps.DefaultRedisCompSpecName).
- AddContainer(corev1.Container{Name: testapps.DefaultRedisContainerName, Image: testapps.DefaultRedisImageName}).
- AddAppInstanceLabel(clusterName).
- AddAppComponentLabel(testapps.DefaultRedisCompSpecName).
- AddAppManangedByLabel().
- AddRoleLabel(role).
- SetReplicas(1).
- Create(&testCtx).GetObject()
-
- By("Creating pods by the StatefulSet")
- testapps.MockReplicationComponentPods(nil, testCtx, sts, clusterName, testapps.DefaultRedisCompSpecName, nil)
- Expect(isStsAndPodsRevisionConsistent(testCtx.Ctx, k8sClient, sts)).Should(BeTrue())
-
- By("Updating the StatefulSet's UpdateRevision")
- sts.Status.UpdateRevision = "new-mock-revision"
- testk8s.PatchStatefulSetStatus(&testCtx, sts.Name, sts.Status)
- podList, err := GetPodListByStatefulSet(ctx, k8sClient, sts)
- Expect(err).To(Succeed())
- Expect(len(podList)).To(Equal(1))
-
- By("Testing get the StatefulSet of the pod")
- ownerSts, err := getPodOwnerReferencesSts(ctx, k8sClient, &podList[0])
- Expect(err).To(Succeed())
- Expect(ownerSts).ShouldNot(BeNil())
-
- By("Deleting the pods of StatefulSet")
- Expect(deleteStsPods(testCtx.Ctx, k8sClient, sts)).Should(Succeed())
- podList, err = GetPodListByStatefulSet(ctx, k8sClient, sts)
- Expect(err).To(Succeed())
- Expect(len(podList)).To(Equal(0))
-
- By("Creating new pods by StatefulSet with new UpdateRevision")
- testapps.MockReplicationComponentPods(nil, testCtx, sts, clusterName, testapps.DefaultRedisCompSpecName, nil)
- Expect(isStsAndPodsRevisionConsistent(testCtx.Ctx, k8sClient, sts)).Should(BeTrue())
- })
- })
-})
diff --git a/internal/common/types.go b/internal/common/types.go
new file mode 100644
index 00000000000..f4279a1c9f4
--- /dev/null
+++ b/internal/common/types.go
@@ -0,0 +1,39 @@
+/*
+Copyright (C) 2022-2023 ApeCloud Co., Ltd
+
+This file is part of KubeBlocks project
+
+This program is free software: you can redistribute it and/or modify
+it under the terms of the GNU Affero General Public License as published by
+the Free Software Foundation, either version 3 of the License, or
+(at your option) any later version.
+
+This program is distributed in the hope that it will be useful
+but WITHOUT ANY WARRANTY; without even the implied warranty of
+MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+GNU Affero General Public License for more details.
+
+You should have received a copy of the GNU Affero General Public License
+along with this program. If not, see .
+*/
+
+package common
+
+// PodRoleNamePair defines a pod name and role name pair.
+type PodRoleNamePair struct {
+ PodName string `json:"podName,omitempty"`
+ RoleName string `json:"roleName,omitempty"`
+}
+
+// GlobalRoleSnapshot defines a global(leader) perspective of all pods role.
+// KB provides two role probe methods: per-pod level role probe and retrieving all node roles from the leader node.
+// The latter is referred to as the global role snapshot. This data structure is used to represent a snapshot of global role information.
+// The snapshot contains two types of information: the mapping relationship between all node names and role names,
+// and the version of the snapshot. The purpose of the snapshot version is to ensure that only role information
+// that is more up-to-date than the current role information on the Pod Label will be updated. This resolves the issue of
+// role information disorder in scenarios such as KB upgrades or exceptions causing restarts,
+// network partitioning leading to split-brain situations, node crashes, and similar occurrences.
+type GlobalRoleSnapshot struct {
+ Version string `json:"term,omitempty"`
+ PodRoleNamePairs []PodRoleNamePair `json:"PodRoleNamePairs,omitempty"`
+}
diff --git a/internal/constant/const.go b/internal/constant/const.go
index 502cd28101b..efc2de22fbc 100644
--- a/internal/constant/const.go
+++ b/internal/constant/const.go
@@ -34,6 +34,7 @@ const (
CfgKeyBackupPVConfigmapName = "BACKUP_PV_CONFIGMAP_NAME" // the configmap containing the persistentVolume template.
CfgKeyBackupPVConfigmapNamespace = "BACKUP_PV_CONFIGMAP_NAMESPACE" // the configmap namespace containing the persistentVolume template.
CfgRecoverVolumeExpansionFailure = "RECOVER_VOLUME_EXPANSION_FAILURE" // refer to feature gates RecoverVolumeExpansionFailure of k8s.
+ CfgKeyProvider = "KUBE_PROVIDER"
// addon config keys
CfgKeyAddonJobTTL = "ADDON_JOB_TTL"
@@ -63,7 +64,6 @@ const (
const (
KBToolsImage = "KUBEBLOCKS_TOOLS_IMAGE"
- KBProbeImage = "KUBEBLOCKS_PROBE_IMAGE"
KBImagePullPolicy = "KUBEBLOCKS_IMAGE_PULL_POLICY"
KBDataScriptClientsImage = "KUBEBLOCKS_DATASCRIPT_CLIENTS_IMAGE"
)
@@ -111,22 +111,19 @@ const (
CMConfigurationConstraintsNameLabelKey = "config.kubeblocks.io/config-constraints-name"
CMConfigurationTemplateVersion = "config.kubeblocks.io/config-template-version"
ConsensusSetAccessModeLabelKey = "cs.apps.kubeblocks.io/access-mode"
- BackupTypeLabelKeyKey = "dataprotection.kubeblocks.io/backup-type"
- DataProtectionLabelBackupNameKey = "dataprotection.kubeblocks.io/backup-name"
AddonNameLabelKey = "extensions.kubeblocks.io/addon-name"
OpsRequestTypeLabelKey = "ops.kubeblocks.io/ops-type"
OpsRequestNameLabelKey = "ops.kubeblocks.io/ops-name"
ServiceDescriptorNameLabelKey = "servicedescriptor.kubeblocks.io/name"
+ RestoreForHScaleLabelKey = "apps.kubeblocks.io/restore-for-hscale"
// kubeblocks.io annotations
- ClusterSnapshotAnnotationKey = "kubeblocks.io/cluster-snapshot" // ClusterSnapshotAnnotationKey saves the snapshot of cluster.
- DefaultClusterVersionAnnotationKey = "kubeblocks.io/is-default-cluster-version" // DefaultClusterVersionAnnotationKey specifies the default cluster version.
- OpsRequestAnnotationKey = "kubeblocks.io/ops-request" // OpsRequestAnnotationKey OpsRequest annotation key in Cluster
- ReconcileAnnotationKey = "kubeblocks.io/reconcile" // ReconcileAnnotationKey Notify k8s object to reconcile
- RestartAnnotationKey = "kubeblocks.io/restart" // RestartAnnotationKey the annotation which notices the StatefulSet/DeploySet to restart
- RestoreFromTimeAnnotationKey = "kubeblocks.io/restore-from-time" // RestoreFromTimeAnnotationKey specifies the time to recover from the backup.
- RestoreFromSrcClusterAnnotationKey = "kubeblocks.io/restore-from-source-cluster" // RestoreFromSrcClusterAnnotationKey specifies the source cluster to recover from the backup.
- RestoreFromBackUpAnnotationKey = "kubeblocks.io/restore-from-backup" // RestoreFromBackUpAnnotationKey specifies the component to recover from the backup.
+ ClusterSnapshotAnnotationKey = "kubeblocks.io/cluster-snapshot" // ClusterSnapshotAnnotationKey saves the snapshot of cluster.
+ DefaultClusterVersionAnnotationKey = "kubeblocks.io/is-default-cluster-version" // DefaultClusterVersionAnnotationKey specifies the default cluster version.
+ OpsRequestAnnotationKey = "kubeblocks.io/ops-request" // OpsRequestAnnotationKey OpsRequest annotation key in Cluster
+ ReconcileAnnotationKey = "kubeblocks.io/reconcile" // ReconcileAnnotationKey Notify k8s object to reconcile
+ RestartAnnotationKey = "kubeblocks.io/restart" // RestartAnnotationKey the annotation which notices the StatefulSet/DeploySet to restart
+ RestoreFromBackupAnnotationKey = "kubeblocks.io/restore-from-backup" // RestoreFromBackupAnnotationKey specifies the component to recover from the backup.
SnapShotForStartAnnotationKey = "kubeblocks.io/snapshot-for-start"
ComponentReplicasAnnotationKey = "apps.kubeblocks.io/component-replicas" // ComponentReplicasAnnotationKey specifies the number of pods in replicas
BackupPolicyTemplateAnnotationKey = "apps.kubeblocks.io/backup-policy-template"
@@ -136,12 +133,6 @@ const (
HaltRecoveryAllowInconsistentResAnnotKey = "clusters.apps.kubeblocks.io/allow-inconsistent-resource"
LeaderAnnotationKey = "cs.apps.kubeblocks.io/leader"
PrimaryAnnotationKey = "rs.apps.kubeblocks.io/primary"
- DefaultBackupPolicyAnnotationKey = "dataprotection.kubeblocks.io/is-default-policy" // DefaultBackupPolicyAnnotationKey specifies the default backup policy.
- DefaultBackupPolicyTemplateAnnotationKey = "dataprotection.kubeblocks.io/is-default-policy-template" // DefaultBackupPolicyTemplateAnnotationKey specifies the default backup policy template.
- DefaultBackupRepoAnnotationKey = "dataprotection.kubeblocks.io/is-default-repo" // DefaultBackupRepoAnnotationKey specifies the default backup repo.
- BackupDataPathPrefixAnnotationKey = "dataprotection.kubeblocks.io/path-prefix" // BackupDataPathPrefixAnnotationKey specifies the backup data path prefix.
- ReconfigureRefAnnotationKey = "dataprotection.kubeblocks.io/reconfigure-ref"
- DataProtectionLabelClusterUIDKey = "dataprotection.kubeblocks.io/cluster-uid"
DisableUpgradeInsConfigurationAnnotationKey = "config.kubeblocks.io/disable-reconfigure"
LastAppliedConfigAnnotationKey = "config.kubeblocks.io/last-applied-configuration"
LastAppliedOpsCRAnnotationKey = "config.kubeblocks.io/last-applied-ops-name"
@@ -151,7 +142,7 @@ const (
ConfigAppliedVersionAnnotationKey = "config.kubeblocks.io/config-applied-version"
KubeBlocksGenerationKey = "kubeblocks.io/generation"
ExtraEnvAnnotationKey = "kubeblocks.io/extra-env"
- LastRoleChangedEventTimestampAnnotationKey = "apps.kubeblocks.io/last-role-changed-event-timestamp"
+ LastRoleSnapshotVersionAnnotationKey = "apps.kubeblocks.io/last-role-snapshot-version"
// kubeblocks.io well-known finalizers
DBClusterFinalizerName = "cluster.kubeblocks.io/finalizer"
@@ -176,6 +167,9 @@ const (
// IgnoreResourceConstraint is used to specify whether to ignore the resource constraint
IgnoreResourceConstraint = "resource.kubeblocks.io/ignore-constraint"
+
+ RBACRoleName = "kubeblocks-cluster-pod-role"
+ RBACClusterRoleName = "kubeblocks-volume-protection-pod-role"
)
const (
@@ -226,22 +220,14 @@ const (
ProbeHTTPPortName = "probe-http-port"
ProbeGRPCPortName = "probe-grpc-port"
ProbeInitContainerName = "kb-initprobe"
- RoleProbeContainerName = "kb-checkrole"
+ WeSyncerContainerName = "kb-we-syncer"
StatusProbeContainerName = "kb-checkstatus"
RunningProbeContainerName = "kb-checkrunning"
VolumeProtectionProbeContainerName = "kb-volume-protection"
// the filedpath name used in event.InvolvedObject.FieldPath
- ProbeCheckRolePath = "spec.containers{" + RoleProbeContainerName + "}"
ProbeCheckStatusPath = "spec.containers{" + StatusProbeContainerName + "}"
ProbeCheckRunningPath = "spec.containers{" + RunningProbeContainerName + "}"
-
- ProbeAgentMountName = "shell2http-mount"
- ProbeAgentMountPath = "/shell2http"
- ProbeAgent = "shell2http"
- ProbeAgentImage = "msoap/shell2http:1.16.0"
- OriginBinaryPath = "/app/shell2http"
- DefaultActionImage = "busybox:latest"
)
const (
@@ -299,29 +285,6 @@ const (
ComponentStatusDefaultPodName = "Unknown"
)
-const (
- // dataProtection env names
- DPTargetPodName = "DP_TARGET_POD_NAME"
- DPDBHost = "DB_HOST" // db host for dataProtection
- DPDBUser = "DB_USER" // db user for dataProtection
- DPDBPassword = "DB_PASSWORD" // db password for dataProtection
- DPBackupDIR = "BACKUP_DIR" // the dest directory for backup data
- DPLogFileDIR = "BACKUP_LOGFILE_DIR" // logfile dir
- DPBackupName = "BACKUP_NAME" // backup cr name
- DPTTL = "TTL" // backup time to live, reference the backupPolicy.spec.retention.ttl
- DPLogfileTTL = "LOGFILE_TTL" // ttl for logfile backup, one more day than backupPolicy.spec.retention.ttl
- DPLogfileTTLSecond = "LOGFILE_TTL_SECOND" // ttl seconds with LOGFILE_TTL, integer format
- DPArchiveInterval = "ARCHIVE_INTERVAL" // archive interval for statefulSet deploy kind, trans from the schedule cronExpression for logfile
- DPBackupInfoFile = "BACKUP_INFO_FILE" // the file name which retains the backup.status info
- DPTimeFormat = "TIME_FORMAT" // golang time format string
- DPVolumeDataDIR = "VOLUME_DATA_DIR" //
- DPKBRecoveryTime = "KB_RECOVERY_TIME" // recovery time
- DPKBRecoveryTimestamp = "KB_RECOVERY_TIMESTAMP" // recovery timestamp
- DPBaseBackupStartTime = "BASE_BACKUP_START_TIME" // base backup start time for pitr
- DPBaseBackupStartTimestamp = "BASE_BACKUP_START_TIMESTAMP" // base backup start timestamp for pitr
- DPBackupStopTime = "BACKUP_STOP_TIME" // backup stop time
-)
-
const (
FeatureGateReplicatedStateMachine = "REPLICATED_STATE_MACHINE" // enable rsm
)
@@ -337,3 +300,10 @@ const (
ServiceDescriptorEndpointKey = "endpoint"
ServiceDescriptorPortKey = "port"
)
+
+const (
+ BackupNameKeyForRestore = "name"
+ BackupNamespaceKeyForRestore = "namespace"
+ VolumeManagementPolicyKeyForRestore = "managementPolicy"
+ RestoreTimeKeyForRestore = "restoreTime"
+)
diff --git a/internal/controller/builder/builder_backup.go b/internal/controller/builder/builder_backup.go
new file mode 100644
index 00000000000..c334b0ac18d
--- /dev/null
+++ b/internal/controller/builder/builder_backup.go
@@ -0,0 +1,49 @@
+/*
+Copyright (C) 2022-2023 ApeCloud Co., Ltd
+
+This file is part of KubeBlocks project
+
+This program is free software: you can redistribute it and/or modify
+it under the terms of the GNU Affero General Public License as published by
+the Free Software Foundation, either version 3 of the License, or
+(at your option) any later version.
+
+This program is distributed in the hope that it will be useful
+but WITHOUT ANY WARRANTY; without even the implied warranty of
+MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+GNU Affero General Public License for more details.
+
+You should have received a copy of the GNU Affero General Public License
+along with this program. If not, see .
+*/
+
+package builder
+
+import (
+ dataprotection "github.com/apecloud/kubeblocks/apis/dataprotection/v1alpha1"
+)
+
+type BackupBuilder struct {
+ BaseBuilder[dataprotection.Backup, *dataprotection.Backup, BackupBuilder]
+}
+
+func NewBackupBuilder(namespace, name string) *BackupBuilder {
+ builder := &BackupBuilder{}
+ builder.init(namespace, name, &dataprotection.Backup{}, builder)
+ return builder
+}
+
+func (builder *BackupBuilder) SetBackupPolicyName(policyName string) *BackupBuilder {
+ builder.get().Spec.BackupPolicyName = policyName
+ return builder
+}
+
+func (builder *BackupBuilder) SetBackupMethod(method string) *BackupBuilder {
+ builder.get().Spec.BackupMethod = method
+ return builder
+}
+
+func (builder *BackupBuilder) SetParentBackupName(parent string) *BackupBuilder {
+ builder.get().Spec.ParentBackupName = parent
+ return builder
+}
diff --git a/internal/controller/builder/builder_backup_test.go b/internal/controller/builder/builder_backup_test.go
new file mode 100644
index 00000000000..e0d2b9ff46f
--- /dev/null
+++ b/internal/controller/builder/builder_backup_test.go
@@ -0,0 +1,45 @@
+/*
+Copyright (C) 2022-2023 ApeCloud Co., Ltd
+
+This file is part of KubeBlocks project
+
+This program is free software: you can redistribute it and/or modify
+it under the terms of the GNU Affero General Public License as published by
+the Free Software Foundation, either version 3 of the License, or
+(at your option) any later version.
+
+This program is distributed in the hope that it will be useful
+but WITHOUT ANY WARRANTY; without even the implied warranty of
+MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+GNU Affero General Public License for more details.
+
+You should have received a copy of the GNU Affero General Public License
+along with this program. If not, see .
+*/
+
+package builder
+
+import (
+ . "github.com/onsi/ginkgo/v2"
+ . "github.com/onsi/gomega"
+)
+
+var _ = Describe("backup builder", func() {
+ It("should work well", func() {
+ const (
+ name = "foo"
+ ns = "default"
+ )
+ policyName := "policyName"
+ backupMethod := "backupMethodName"
+ backup := NewBackupBuilder(ns, name).
+ SetBackupPolicyName(policyName).
+ SetBackupMethod(backupMethod).
+ GetObject()
+
+ Expect(backup.Name).Should(Equal(name))
+ Expect(backup.Namespace).Should(Equal(ns))
+ Expect(backup.Spec.BackupPolicyName).Should(Equal(policyName))
+ Expect(backup.Spec.BackupMethod).Should(Equal(backupMethod))
+ })
+})
diff --git a/internal/controller/builder/builder_cluster_role_binding.go b/internal/controller/builder/builder_cluster_role_binding.go
new file mode 100644
index 00000000000..55db20863bc
--- /dev/null
+++ b/internal/controller/builder/builder_cluster_role_binding.go
@@ -0,0 +1,44 @@
+/*
+Copyright (C) 2022-2023 ApeCloud Co., Ltd
+
+This file is part of KubeBlocks project
+
+This program is free software: you can redistribute it and/or modify
+it under the terms of the GNU Affero General Public License as published by
+the Free Software Foundation, either version 3 of the License, or
+(at your option) any later version.
+
+This program is distributed in the hope that it will be useful
+but WITHOUT ANY WARRANTY; without even the implied warranty of
+MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+GNU Affero General Public License for more details.
+
+You should have received a copy of the GNU Affero General Public License
+along with this program. If not, see .
+*/
+
+package builder
+
+import (
+ rbacv1 "k8s.io/api/rbac/v1"
+)
+
+type ClusterRoleBindingBuilder struct {
+ BaseBuilder[rbacv1.ClusterRoleBinding, *rbacv1.ClusterRoleBinding, ClusterRoleBindingBuilder]
+}
+
+func NewClusterRoleBindingBuilder(namespace, name string) *ClusterRoleBindingBuilder {
+ builder := &ClusterRoleBindingBuilder{}
+ builder.init(namespace, name, &rbacv1.ClusterRoleBinding{}, builder)
+ return builder
+}
+
+func (builder *ClusterRoleBindingBuilder) SetRoleRef(roleRef rbacv1.RoleRef) *ClusterRoleBindingBuilder {
+ builder.get().RoleRef = roleRef
+ return builder
+}
+
+func (builder *ClusterRoleBindingBuilder) AddSubjects(subjects ...rbacv1.Subject) *ClusterRoleBindingBuilder {
+ builder.get().Subjects = append(builder.get().Subjects, subjects...)
+ return builder
+}
diff --git a/internal/controller/builder/builder_cluster_role_binding_test.go b/internal/controller/builder/builder_cluster_role_binding_test.go
new file mode 100644
index 00000000000..5ef10f07810
--- /dev/null
+++ b/internal/controller/builder/builder_cluster_role_binding_test.go
@@ -0,0 +1,60 @@
+/*
+Copyright (C) 2022-2023 ApeCloud Co., Ltd
+
+This file is part of KubeBlocks project
+
+This program is free software: you can redistribute it and/or modify
+it under the terms of the GNU Affero General Public License as published by
+the Free Software Foundation, either version 3 of the License, or
+(at your option) any later version.
+
+This program is distributed in the hope that it will be useful
+but WITHOUT ANY WARRANTY; without even the implied warranty of
+MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+GNU Affero General Public License for more details.
+
+You should have received a copy of the GNU Affero General Public License
+along with this program. If not, see .
+*/
+
+package builder
+
+import (
+ "fmt"
+
+ . "github.com/onsi/ginkgo/v2"
+ . "github.com/onsi/gomega"
+
+ rbacv1 "k8s.io/api/rbac/v1"
+
+ "github.com/apecloud/kubeblocks/internal/constant"
+)
+
+var _ = Describe("cluster role binding builder", func() {
+ It("should work well", func() {
+ const (
+ name = "foo"
+ ns = "default"
+ )
+ roleRef := rbacv1.RoleRef{
+ APIGroup: rbacv1.GroupName,
+ Kind: "ClusterRole",
+ Name: constant.RBACRoleName,
+ }
+ subject := rbacv1.Subject{
+ Kind: rbacv1.ServiceAccountKind,
+ Namespace: ns,
+ Name: fmt.Sprintf("kb-%s", name),
+ }
+ clusterRoleBinding := NewClusterRoleBindingBuilder(ns, name).
+ SetRoleRef(roleRef).
+ AddSubjects(subject).
+ GetObject()
+
+ Expect(clusterRoleBinding.Name).Should(Equal(name))
+ Expect(clusterRoleBinding.Namespace).Should(Equal(ns))
+ Expect(clusterRoleBinding.RoleRef).Should(Equal(roleRef))
+ Expect(clusterRoleBinding.Subjects).Should(HaveLen(1))
+ Expect(clusterRoleBinding.Subjects[0]).Should(Equal(subject))
+ })
+})
diff --git a/internal/controller/builder/builder_configuration.go b/internal/controller/builder/builder_configuration.go
index d7ac8acb893..b065231fe16 100644
--- a/internal/controller/builder/builder_configuration.go
+++ b/internal/controller/builder/builder_configuration.go
@@ -43,17 +43,11 @@ func (c *ConfigurationBuilder) Component(component string) *ConfigurationBuilder
return c
}
-func (c *ConfigurationBuilder) ClusterVerRef(clusterVer string) *ConfigurationBuilder {
- c.get().Spec.ClusterVersionRef = clusterVer
- return c
-}
-
-func (c *ConfigurationBuilder) ClusterDefRef(clusterDef string) *ConfigurationBuilder {
- c.get().Spec.ClusterDefRef = clusterDef
- return c
-}
-
-func (c *ConfigurationBuilder) AddConfigurationItem(name string) *ConfigurationBuilder {
- c.get().Spec.ConfigItemDetails = append(c.get().Spec.ConfigItemDetails, v1alpha1.ConfigurationItemDetail{Name: name})
+func (c *ConfigurationBuilder) AddConfigurationItem(configSpec v1alpha1.ComponentConfigSpec) *ConfigurationBuilder {
+ c.get().Spec.ConfigItemDetails = append(c.get().Spec.ConfigItemDetails,
+ v1alpha1.ConfigurationItemDetail{
+ Name: configSpec.Name,
+ ConfigSpec: configSpec.DeepCopy(),
+ })
return c
}
diff --git a/internal/controller/builder/builder_configuration_test.go b/internal/controller/builder/builder_configuration_test.go
index a6b6bede42f..ac1f2fba2d2 100644
--- a/internal/controller/builder/builder_configuration_test.go
+++ b/internal/controller/builder/builder_configuration_test.go
@@ -23,33 +23,36 @@ import (
. "github.com/onsi/ginkgo/v2"
. "github.com/onsi/gomega"
+ "github.com/apecloud/kubeblocks/apis/apps/v1alpha1"
"github.com/apecloud/kubeblocks/internal/configuration/core"
)
var _ = Describe("configuration builder", func() {
It("should work well", func() {
const (
- clusterName = "test"
- componentName = "mysql"
- clusterDefName = "mysql-cd"
- clusterVerName = "mysql-cv"
- ns = "default"
+ clusterName = "test"
+ componentName = "mysql"
+ ns = "default"
)
name := core.GenerateComponentConfigurationName(clusterName, componentName)
config := NewConfigurationBuilder(ns, name).
ClusterRef(clusterName).
Component(componentName).
- ClusterDefRef(clusterDefName).
- ClusterVerRef(clusterVerName).
- AddConfigurationItem("mysql-config").
- AddConfigurationItem("mysql-oteld-config").
+ AddConfigurationItem(v1alpha1.ComponentConfigSpec{
+ ComponentTemplateSpec: v1alpha1.ComponentTemplateSpec{
+ Name: "mysql-config",
+ },
+ }).
+ AddConfigurationItem(v1alpha1.ComponentConfigSpec{
+ ComponentTemplateSpec: v1alpha1.ComponentTemplateSpec{
+ Name: "mysql-oteld-config",
+ },
+ }).
GetObject()
Expect(config.Name).Should(BeEquivalentTo(name))
Expect(config.Spec.ClusterRef).Should(BeEquivalentTo(clusterName))
Expect(config.Spec.ComponentName).Should(BeEquivalentTo(componentName))
- Expect(config.Spec.ClusterVersionRef).Should(BeEquivalentTo(clusterVerName))
- Expect(config.Spec.ClusterDefRef).Should(BeEquivalentTo(clusterDefName))
Expect(len(config.Spec.ConfigItemDetails)).Should(Equal(2))
})
})
diff --git a/internal/controller/builder/builder_container.go b/internal/controller/builder/builder_container.go
new file mode 100644
index 00000000000..fc483fa2d93
--- /dev/null
+++ b/internal/controller/builder/builder_container.go
@@ -0,0 +1,102 @@
+/*
+Copyright (C) 2022-2023 ApeCloud Co., Ltd
+
+This file is part of KubeBlocks project
+
+This program is free software: you can redistribute it and/or modify
+it under the terms of the GNU Affero General Public License as published by
+the Free Software Foundation, either version 3 of the License, or
+(at your option) any later version.
+
+This program is distributed in the hope that it will be useful
+but WITHOUT ANY WARRANTY; without even the implied warranty of
+MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+GNU Affero General Public License for more details.
+
+You should have received a copy of the GNU Affero General Public License
+along with this program. If not, see .
+*/
+
+package builder
+
+import (
+ corev1 "k8s.io/api/core/v1"
+)
+
+type ContainerBuilder struct {
+ object *corev1.Container
+}
+
+func NewContainerBuilder(name string) *ContainerBuilder {
+ builder := &ContainerBuilder{}
+ builder.init(name, &corev1.Container{})
+ return builder
+}
+
+func (builder *ContainerBuilder) init(name string, obj *corev1.Container) {
+ obj.Name = name
+ builder.object = obj
+}
+
+func (builder *ContainerBuilder) get() *corev1.Container {
+ return builder.object
+}
+
+func (builder *ContainerBuilder) GetObject() *corev1.Container {
+ return builder.object
+}
+
+func (builder *ContainerBuilder) AddCommands(commands ...string) *ContainerBuilder {
+ builder.get().Command = append(builder.get().Command, commands...)
+ return builder
+}
+
+func (builder *ContainerBuilder) AddArgs(args ...string) *ContainerBuilder {
+ builder.get().Args = append(builder.get().Args, args...)
+ return builder
+}
+
+func (builder *ContainerBuilder) AddEnv(env ...corev1.EnvVar) *ContainerBuilder {
+ builder.get().Env = append(builder.get().Env, env...)
+ return builder
+}
+
+func (builder *ContainerBuilder) SetImage(image string) *ContainerBuilder {
+ builder.get().Image = image
+ return builder
+}
+
+func (builder *ContainerBuilder) SetImagePullPolicy(policy corev1.PullPolicy) *ContainerBuilder {
+ builder.get().ImagePullPolicy = policy
+ return builder
+}
+
+func (builder *ContainerBuilder) AddVolumeMounts(mounts ...corev1.VolumeMount) *ContainerBuilder {
+ builder.get().VolumeMounts = append(builder.get().VolumeMounts, mounts...)
+ return builder
+}
+
+func (builder *ContainerBuilder) SetSecurityContext(ctx corev1.SecurityContext) *ContainerBuilder {
+ builder.get().SecurityContext = &ctx
+ return builder
+}
+
+func (builder *ContainerBuilder) SetResources(resources corev1.ResourceRequirements) *ContainerBuilder {
+ builder.get().Resources = resources
+ return builder
+}
+
+func (builder *ContainerBuilder) AddPorts(ports ...corev1.ContainerPort) *ContainerBuilder {
+ builder.get().Ports = append(builder.get().Ports, ports...)
+ return builder
+}
+
+func (builder *ContainerBuilder) SetReadinessProbe(probe corev1.Probe) *ContainerBuilder {
+ builder.get().ReadinessProbe = &probe
+ return builder
+}
+
+func (builder *ContainerBuilder) SetStartupProbe(probe corev1.Probe) *ContainerBuilder {
+ builder.get().StartupProbe = &probe
+ return builder
+}
diff --git a/internal/controller/builder/builder_container_test.go b/internal/controller/builder/builder_container_test.go
new file mode 100644
index 00000000000..2785d3c8990
--- /dev/null
+++ b/internal/controller/builder/builder_container_test.go
@@ -0,0 +1,144 @@
+/*
+Copyright (C) 2022-2023 ApeCloud Co., Ltd
+
+This file is part of KubeBlocks project
+
+This program is free software: you can redistribute it and/or modify
+it under the terms of the GNU Affero General Public License as published by
+the Free Software Foundation, either version 3 of the License, or
+(at your option) any later version.
+
+This program is distributed in the hope that it will be useful
+but WITHOUT ANY WARRANTY; without even the implied warranty of
+MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+GNU Affero General Public License for more details.
+
+You should have received a copy of the GNU Affero General Public License
+along with this program. If not, see .
+*/
+
+package builder
+
+import (
+ . "github.com/onsi/ginkgo/v2"
+ . "github.com/onsi/gomega"
+
+ corev1 "k8s.io/api/core/v1"
+ "k8s.io/apimachinery/pkg/api/resource"
+ "k8s.io/apimachinery/pkg/util/intstr"
+)
+
+var _ = Describe("container builder", func() {
+ It("should work well", func() {
+ const name = "foo"
+ commands := []string{
+ name,
+ "--bar",
+ }
+ args := []string{
+ "arg1",
+ "arg2",
+ }
+ env := []corev1.EnvVar{
+ {
+ Name: name,
+ Value: "bar",
+ },
+ {
+ Name: "hello",
+ ValueFrom: &corev1.EnvVarSource{
+ FieldRef: &corev1.ObjectFieldSelector{
+ FieldPath: "metadata.name",
+ },
+ },
+ },
+ }
+ image := "foo:latest"
+ policy := corev1.PullAlways
+ mounts := []corev1.VolumeMount{
+ {
+ Name: name,
+ MountPath: "/data/foo",
+ },
+ {
+ Name: "bar",
+ ReadOnly: true,
+ MountPath: "/log/bar",
+ },
+ }
+ user := int64(0)
+ ctx := corev1.SecurityContext{
+ RunAsUser: &user,
+ }
+
+ resourceQuantityValue := func(value string) resource.Quantity {
+ quantity, _ := resource.ParseQuantity(value)
+ return quantity
+ }
+ resources := corev1.ResourceRequirements{
+ Limits: map[corev1.ResourceName]resource.Quantity{
+ corev1.ResourceCPU: resourceQuantityValue("0.5"),
+ corev1.ResourceMemory: resourceQuantityValue("500m"),
+ },
+ Requests: map[corev1.ResourceName]resource.Quantity{
+ corev1.ResourceCPU: resourceQuantityValue("0.5"),
+ corev1.ResourceMemory: resourceQuantityValue("500m"),
+ },
+ }
+ ports := []corev1.ContainerPort{
+ {
+ Name: name,
+ ContainerPort: 12345,
+ Protocol: corev1.ProtocolTCP,
+ },
+ {
+ Name: "bar",
+ ContainerPort: 54321,
+ Protocol: corev1.ProtocolUDP,
+ },
+ }
+ readinessProbe := corev1.Probe{
+ ProbeHandler: corev1.ProbeHandler{
+ Exec: &corev1.ExecAction{
+ Command: []string{},
+ },
+ },
+ }
+ startupProbe := corev1.Probe{
+ ProbeHandler: corev1.ProbeHandler{
+ TCPSocket: &corev1.TCPSocketAction{
+ Port: intstr.FromInt(12345),
+ },
+ },
+ }
+ container := NewContainerBuilder(name).
+ AddCommands(commands...).
+ AddArgs(args...).
+ AddEnv(env...).
+ SetImage(image).
+ SetImagePullPolicy(policy).
+ AddVolumeMounts(mounts...).
+ SetSecurityContext(ctx).
+ SetResources(resources).
+ AddPorts(ports...).
+ SetReadinessProbe(readinessProbe).
+ SetStartupProbe(startupProbe).
+ GetObject()
+
+ Expect(container.Name).Should(Equal(name))
+ Expect(container.Command).Should(Equal(commands))
+ Expect(container.Args).Should(Equal(args))
+ Expect(container.Env).Should(Equal(env))
+ Expect(container.Image).Should(Equal(image))
+ Expect(container.ImagePullPolicy).Should(Equal(policy))
+ Expect(container.VolumeMounts).Should(Equal(mounts))
+ Expect(container.SecurityContext).ShouldNot(BeNil())
+ Expect(*container.SecurityContext).Should(Equal(ctx))
+ Expect(container.Resources).Should(Equal(resources))
+ Expect(container.Ports).Should(Equal(ports))
+ Expect(container.ReadinessProbe).ShouldNot(BeNil())
+ Expect(*container.ReadinessProbe).Should(Equal(readinessProbe))
+ Expect(container.StartupProbe).ShouldNot(BeNil())
+ Expect(*container.StartupProbe).Should(Equal(startupProbe))
+ })
+})
diff --git a/internal/controller/builder/builder_job.go b/internal/controller/builder/builder_job.go
index 8c8d93dd1e9..5992857598c 100644
--- a/internal/controller/builder/builder_job.go
+++ b/internal/controller/builder/builder_job.go
@@ -56,3 +56,13 @@ func (builder *JobBuilder) SetSuspend(suspend bool) *JobBuilder {
builder.get().Spec.Suspend = &suspend
return builder
}
+
+func (builder *JobBuilder) SetBackoffLimit(limit int32) *JobBuilder {
+ builder.get().Spec.BackoffLimit = &limit
+ return builder
+}
+
+func (builder *JobBuilder) SetTTLSecondsAfterFinished(ttl int32) *JobBuilder {
+ builder.get().Spec.TTLSecondsAfterFinished = &ttl
+ return builder
+}
diff --git a/internal/controller/builder/builder_job_test.go b/internal/controller/builder/builder_job_test.go
index fcc48fd4212..5f70ce9136c 100644
--- a/internal/controller/builder/builder_job_test.go
+++ b/internal/controller/builder/builder_job_test.go
@@ -51,10 +51,14 @@ var _ = Describe("job builder", func() {
}
selectorKey, selectorValue := "foo", "bar"
suspend := true
+ limit := int32(5)
+ ttl := int32(12)
job := NewJobBuilder(ns, name).
SetPodTemplateSpec(template).
AddSelector(selectorKey, selectorValue).
SetSuspend(suspend).
+ SetBackoffLimit(limit).
+ SetTTLSecondsAfterFinished(ttl).
GetObject()
Expect(job.Name).Should(Equal(name))
@@ -65,5 +69,9 @@ var _ = Describe("job builder", func() {
Expect(job.Spec.Selector.MatchLabels[selectorKey]).Should(Equal(selectorValue))
Expect(job.Spec.Suspend).ShouldNot(BeNil())
Expect(*job.Spec.Suspend).Should(Equal(suspend))
+ Expect(job.Spec.BackoffLimit).ShouldNot(BeNil())
+ Expect(*job.Spec.BackoffLimit).Should(Equal(limit))
+ Expect(job.Spec.TTLSecondsAfterFinished).ShouldNot(BeNil())
+ Expect(*job.Spec.TTLSecondsAfterFinished).Should(Equal(ttl))
})
})
diff --git a/internal/controller/builder/builder_pdb.go b/internal/controller/builder/builder_pdb.go
new file mode 100644
index 00000000000..d91fd5409ac
--- /dev/null
+++ b/internal/controller/builder/builder_pdb.go
@@ -0,0 +1,71 @@
+/*
+Copyright (C) 2022-2023 ApeCloud Co., Ltd
+
+This file is part of KubeBlocks project
+
+This program is free software: you can redistribute it and/or modify
+it under the terms of the GNU Affero General Public License as published by
+the Free Software Foundation, either version 3 of the License, or
+(at your option) any later version.
+
+This program is distributed in the hope that it will be useful
+but WITHOUT ANY WARRANTY; without even the implied warranty of
+MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+GNU Affero General Public License for more details.
+
+You should have received a copy of the GNU Affero General Public License
+along with this program. If not, see .
+*/
+
+package builder
+
+import (
+ policyv1 "k8s.io/api/policy/v1"
+ metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
+ "k8s.io/apimachinery/pkg/util/intstr"
+)
+
+type PDBBuilder struct {
+ BaseBuilder[policyv1.PodDisruptionBudget, *policyv1.PodDisruptionBudget, PDBBuilder]
+}
+
+func NewPDBBuilder(namespace, name string) *PDBBuilder {
+ builder := &PDBBuilder{}
+ builder.init(namespace, name, &policyv1.PodDisruptionBudget{}, builder)
+ return builder
+}
+
+func (builder *PDBBuilder) SetMinAvailable(minAvailable intstr.IntOrString) *PDBBuilder {
+ builder.get().Spec.MinAvailable = &minAvailable
+ return builder
+}
+
+func (builder *PDBBuilder) AddSelector(key, value string) *PDBBuilder {
+ selector := builder.get().Spec.Selector
+ if selector == nil {
+ selector = &metav1.LabelSelector{
+ MatchLabels: map[string]string{},
+ }
+ }
+ selector.MatchLabels[key] = value
+ builder.get().Spec.Selector = selector
+ return builder
+}
+
+func (builder *PDBBuilder) AddSelectors(keyValues ...string) *PDBBuilder {
+ return builder.AddSelectorsInMap(WithMap(keyValues...))
+}
+
+func (builder *PDBBuilder) AddSelectorsInMap(keyValues map[string]string) *PDBBuilder {
+ selector := builder.get().Spec.Selector
+ if selector == nil {
+ selector = &metav1.LabelSelector{
+ MatchLabels: map[string]string{},
+ }
+ }
+ for k, v := range keyValues {
+ selector.MatchLabels[k] = v
+ }
+ builder.get().Spec.Selector = selector
+ return builder
+}
diff --git a/internal/controller/builder/builder_pdb_test.go b/internal/controller/builder/builder_pdb_test.go
new file mode 100644
index 00000000000..87709ab0f01
--- /dev/null
+++ b/internal/controller/builder/builder_pdb_test.go
@@ -0,0 +1,60 @@
+/*
+Copyright (C) 2022-2023 ApeCloud Co., Ltd
+
+This file is part of KubeBlocks project
+
+This program is free software: you can redistribute it and/or modify
+it under the terms of the GNU Affero General Public License as published by
+the Free Software Foundation, either version 3 of the License, or
+(at your option) any later version.
+
+This program is distributed in the hope that it will be useful
+but WITHOUT ANY WARRANTY; without even the implied warranty of
+MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+GNU Affero General Public License for more details.
+
+You should have received a copy of the GNU Affero General Public License
+along with this program. If not, see .
+*/
+
+package builder
+
+import (
+ . "github.com/onsi/ginkgo/v2"
+ . "github.com/onsi/gomega"
+ "k8s.io/apimachinery/pkg/util/intstr"
+)
+
+var _ = Describe("pdb builder", func() {
+ It("should work well", func() {
+ const (
+ name = "foo"
+ ns = "default"
+ selectorKey1, selectorValue1 = "foo-1", "bar-1"
+ selectorKey2, selectorValue2 = "foo-2", "bar-2"
+ selectorKey3, selectorValue3 = "foo-3", "bar-3"
+ selectorKey4, selectorValue4 = "foo-4", "bar-4"
+ )
+ selectors := map[string]string{selectorKey4: selectorValue4}
+ minAvailable := intstr.FromInt(3)
+ pdb := NewPDBBuilder(ns, name).
+ AddSelector(selectorKey1, selectorValue1).
+ AddSelectors(selectorKey2, selectorValue2, selectorKey3, selectorValue3).
+ AddSelectorsInMap(selectors).
+ SetMinAvailable(minAvailable).
+ GetObject()
+
+ Expect(pdb.Name).Should(Equal(name))
+ Expect(pdb.Namespace).Should(Equal(ns))
+ Expect(pdb.Spec.Selector).ShouldNot(BeNil())
+ Expect(pdb.Spec.Selector.MatchLabels).ShouldNot(BeNil())
+ Expect(pdb.Spec.Selector).ShouldNot(BeNil())
+ Expect(pdb.Spec.Selector.MatchLabels).Should(HaveLen(4))
+ Expect(pdb.Spec.Selector.MatchLabels[selectorKey1]).Should(Equal(selectorValue1))
+ Expect(pdb.Spec.Selector.MatchLabels[selectorKey2]).Should(Equal(selectorValue2))
+ Expect(pdb.Spec.Selector.MatchLabels[selectorKey3]).Should(Equal(selectorValue3))
+ Expect(pdb.Spec.Selector.MatchLabels[selectorKey4]).Should(Equal(selectorValue4))
+ Expect(pdb.Spec.MinAvailable).ShouldNot(BeNil())
+ Expect(*pdb.Spec.MinAvailable).Should(Equal(minAvailable))
+ })
+})
diff --git a/internal/controller/builder/builder_pod.go b/internal/controller/builder/builder_pod.go
index 04b32f31ca6..39b4f9dfa35 100644
--- a/internal/controller/builder/builder_pod.go
+++ b/internal/controller/builder/builder_pod.go
@@ -42,3 +42,23 @@ func (builder *PodBuilder) AddContainer(container corev1.Container) *PodBuilder
builder.get().Spec.Containers = containers
return builder
}
+
+func (builder *PodBuilder) AddVolumes(volumes ...corev1.Volume) *PodBuilder {
+ builder.get().Spec.Volumes = append(builder.get().Spec.Volumes, volumes...)
+ return builder
+}
+
+func (builder *PodBuilder) SetRestartPolicy(policy corev1.RestartPolicy) *PodBuilder {
+ builder.get().Spec.RestartPolicy = policy
+ return builder
+}
+
+func (builder *PodBuilder) SetSecurityContext(ctx corev1.PodSecurityContext) *PodBuilder {
+ builder.get().Spec.SecurityContext = &ctx
+ return builder
+}
+
+func (builder *PodBuilder) AddTolerations(tolerations ...corev1.Toleration) *PodBuilder {
+ builder.get().Spec.Tolerations = append(builder.get().Spec.Tolerations, tolerations...)
+ return builder
+}
diff --git a/internal/controller/builder/builder_pod_test.go b/internal/controller/builder/builder_pod_test.go
index 402715bdfed..be7ecf6dbdc 100644
--- a/internal/controller/builder/builder_pod_test.go
+++ b/internal/controller/builder/builder_pod_test.go
@@ -31,33 +31,48 @@ var _ = Describe("pod builder", func() {
name := "foo"
ns := "default"
port := int32(12345)
- container := corev1.Container{
- Name: "foo-1",
- Image: "bar-2",
- Ports: []corev1.ContainerPort{
- {
- Name: "foo-1",
+ container := *NewContainerBuilder("foo-1").
+ SetImage("bar-1").
+ AddPorts(corev1.ContainerPort{
+ Name: "foo-1",
+ Protocol: corev1.ProtocolTCP,
+ ContainerPort: port,
+ }).GetObject()
+ containers := []corev1.Container{
+ *NewContainerBuilder("foo-2").SetImage("bar-2").
+ AddPorts(corev1.ContainerPort{
+ Name: "foo-2",
Protocol: corev1.ProtocolTCP,
ContainerPort: port,
+ }).GetObject(),
+ }
+ volumes := []corev1.Volume{
+ {
+ Name: "data",
+ VolumeSource: corev1.VolumeSource{
+ EmptyDir: &corev1.EmptyDirVolumeSource{},
},
},
}
- containers := []corev1.Container{
+ restartPolicy := corev1.RestartPolicyOnFailure
+ user := int64(0)
+ ctx := corev1.PodSecurityContext{
+ RunAsUser: &user,
+ }
+ tolerations := []corev1.Toleration{
{
- Name: "foo-2",
- Image: "bar-2",
- Ports: []corev1.ContainerPort{
- {
- Name: "foo-2",
- Protocol: corev1.ProtocolTCP,
- ContainerPort: port,
- },
- },
+ Key: "node",
+ Operator: corev1.TolerationOpEqual,
+ Value: "node-0",
},
}
pod := NewPodBuilder(ns, name).
SetContainers(containers).
AddContainer(container).
+ AddVolumes(volumes...).
+ SetRestartPolicy(restartPolicy).
+ SetSecurityContext(ctx).
+ AddTolerations(tolerations...).
GetObject()
Expect(pod.Name).Should(Equal(name))
@@ -65,5 +80,12 @@ var _ = Describe("pod builder", func() {
Expect(pod.Spec.Containers).Should(HaveLen(2))
Expect(pod.Spec.Containers[0]).Should(Equal(containers[0]))
Expect(pod.Spec.Containers[1]).Should(Equal(container))
+ Expect(pod.Spec.Volumes).Should(HaveLen(1))
+ Expect(pod.Spec.Volumes[0]).Should(Equal(volumes[0]))
+ Expect(pod.Spec.RestartPolicy).Should(Equal(restartPolicy))
+ Expect(pod.Spec.SecurityContext).ShouldNot(BeNil())
+ Expect(*pod.Spec.SecurityContext).Should(Equal(ctx))
+ Expect(pod.Spec.Tolerations).Should(HaveLen(1))
+ Expect(pod.Spec.Tolerations[0]).Should(Equal(tolerations[0]))
})
})
diff --git a/internal/controller/builder/builder_pvc.go b/internal/controller/builder/builder_pvc.go
index 0f2f6cb79ff..9b4c82fecbe 100644
--- a/internal/controller/builder/builder_pvc.go
+++ b/internal/controller/builder/builder_pvc.go
@@ -35,3 +35,18 @@ func (builder *PVCBuilder) SetResources(resources corev1.ResourceRequirements) *
builder.get().Spec.Resources = resources
return builder
}
+
+func (builder *PVCBuilder) SetAccessModes(accessModes []corev1.PersistentVolumeAccessMode) *PVCBuilder {
+ builder.get().Spec.AccessModes = accessModes
+ return builder
+}
+
+func (builder *PVCBuilder) SetStorageClass(sc string) *PVCBuilder {
+ builder.get().Spec.StorageClassName = &sc
+ return builder
+}
+
+func (builder *PVCBuilder) SetDataSource(dataSource corev1.TypedLocalObjectReference) *PVCBuilder {
+ builder.get().Spec.DataSource = &dataSource
+ return builder
+}
diff --git a/internal/controller/builder/builder_pvc_test.go b/internal/controller/builder/builder_pvc_test.go
index 518fe0b1e5b..2ee9585d01c 100644
--- a/internal/controller/builder/builder_pvc_test.go
+++ b/internal/controller/builder/builder_pvc_test.go
@@ -38,12 +38,30 @@ var _ = Describe("pvc builder", func() {
"CPU": resource.MustParse("500m"),
},
}
+ accessModes := []corev1.PersistentVolumeAccessMode{
+ corev1.ReadWriteOnce,
+ }
+ sc := "openebs-local-pv"
+ apiGroup := "apps.kubeblocks.io/v1alpha1"
+ dataSource := corev1.TypedLocalObjectReference{
+ APIGroup: &apiGroup,
+ Kind: "Backup",
+ Name: "cluster-component-backup",
+ }
pvc := NewPVCBuilder(ns, name).
SetResources(resources).
+ SetAccessModes(accessModes).
+ SetStorageClass(sc).
+ SetDataSource(dataSource).
GetObject()
Expect(pvc.Name).Should(Equal(name))
Expect(pvc.Namespace).Should(Equal(ns))
Expect(pvc.Spec.Resources).Should(Equal(resources))
+ Expect(pvc.Spec.AccessModes).Should(Equal(accessModes))
+ Expect(pvc.Spec.StorageClassName).ShouldNot(BeNil())
+ Expect(*pvc.Spec.StorageClassName).Should(Equal(sc))
+ Expect(pvc.Spec.DataSource).ShouldNot(BeNil())
+ Expect(*pvc.Spec.DataSource).Should(Equal(dataSource))
})
})
diff --git a/controllers/apps/components/stateless_workload.go b/internal/controller/builder/builder_role_binding.go
similarity index 52%
rename from controllers/apps/components/stateless_workload.go
rename to internal/controller/builder/builder_role_binding.go
index f091a9bc70f..2a47997dfc5 100644
--- a/controllers/apps/components/stateless_workload.go
+++ b/internal/controller/builder/builder_role_binding.go
@@ -17,27 +17,28 @@ You should have received a copy of the GNU Affero General Public License
along with this program. If not, see .
*/
-package components
+package builder
import (
- "github.com/apecloud/kubeblocks/internal/controller/factory"
- "sigs.k8s.io/controller-runtime/pkg/client"
+ rbacv1 "k8s.io/api/rbac/v1"
)
-type statelessComponentWorkloadBuilder struct {
- componentWorkloadBuilderBase
+type RoleBindingBuilder struct {
+ BaseBuilder[rbacv1.RoleBinding, *rbacv1.RoleBinding, RoleBindingBuilder]
}
-var _ componentWorkloadBuilder = &statelessComponentWorkloadBuilder{}
-
-func (b *statelessComponentWorkloadBuilder) BuildWorkload() componentWorkloadBuilder {
- buildfn := func() ([]client.Object, error) {
- deploy, err := factory.BuildDeploy(b.ReqCtx, b.Comp.GetCluster(), b.Comp.GetSynthesizedComponent(), b.EnvConfig.Name)
- if err != nil {
- return nil, err
- }
- b.Workload = deploy
- return nil, nil // don't return deployment here
- }
- return b.BuildWrapper(buildfn)
+func NewRoleBindingBuilder(namespace, name string) *RoleBindingBuilder {
+ builder := &RoleBindingBuilder{}
+ builder.init(namespace, name, &rbacv1.RoleBinding{}, builder)
+ return builder
+}
+
+func (builder *RoleBindingBuilder) SetRoleRef(roleRef rbacv1.RoleRef) *RoleBindingBuilder {
+ builder.get().RoleRef = roleRef
+ return builder
+}
+
+func (builder *RoleBindingBuilder) AddSubjects(subjects ...rbacv1.Subject) *RoleBindingBuilder {
+ builder.get().Subjects = append(builder.get().Subjects, subjects...)
+ return builder
}
diff --git a/internal/controller/builder/builder_role_binding_test.go b/internal/controller/builder/builder_role_binding_test.go
new file mode 100644
index 00000000000..b621bbcccf1
--- /dev/null
+++ b/internal/controller/builder/builder_role_binding_test.go
@@ -0,0 +1,60 @@
+/*
+Copyright (C) 2022-2023 ApeCloud Co., Ltd
+
+This file is part of KubeBlocks project
+
+This program is free software: you can redistribute it and/or modify
+it under the terms of the GNU Affero General Public License as published by
+the Free Software Foundation, either version 3 of the License, or
+(at your option) any later version.
+
+This program is distributed in the hope that it will be useful
+but WITHOUT ANY WARRANTY; without even the implied warranty of
+MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+GNU Affero General Public License for more details.
+
+You should have received a copy of the GNU Affero General Public License
+along with this program. If not, see .
+*/
+
+package builder
+
+import (
+ "fmt"
+
+ . "github.com/onsi/ginkgo/v2"
+ . "github.com/onsi/gomega"
+
+ rbacv1 "k8s.io/api/rbac/v1"
+
+ "github.com/apecloud/kubeblocks/internal/constant"
+)
+
+var _ = Describe("role binding builder", func() {
+ It("should work well", func() {
+ const (
+ name = "foo"
+ ns = "default"
+ )
+ roleRef := rbacv1.RoleRef{
+ APIGroup: rbacv1.GroupName,
+ Kind: "ClusterRole",
+ Name: constant.RBACRoleName,
+ }
+ subject := rbacv1.Subject{
+ Kind: rbacv1.ServiceAccountKind,
+ Namespace: ns,
+ Name: fmt.Sprintf("kb-%s", name),
+ }
+ roleBinding := NewRoleBindingBuilder(ns, name).
+ SetRoleRef(roleRef).
+ AddSubjects(subject).
+ GetObject()
+
+ Expect(roleBinding.Name).Should(Equal(name))
+ Expect(roleBinding.Namespace).Should(Equal(ns))
+ Expect(roleBinding.RoleRef).Should(Equal(roleRef))
+ Expect(roleBinding.Subjects).Should(HaveLen(1))
+ Expect(roleBinding.Subjects[0]).Should(Equal(subject))
+ })
+})
diff --git a/internal/controller/builder/builder_service_account.go b/internal/controller/builder/builder_service_account.go
new file mode 100644
index 00000000000..070e7606f64
--- /dev/null
+++ b/internal/controller/builder/builder_service_account.go
@@ -0,0 +1,34 @@
+/*
+Copyright (C) 2022-2023 ApeCloud Co., Ltd
+
+This file is part of KubeBlocks project
+
+This program is free software: you can redistribute it and/or modify
+it under the terms of the GNU Affero General Public License as published by
+the Free Software Foundation, either version 3 of the License, or
+(at your option) any later version.
+
+This program is distributed in the hope that it will be useful
+but WITHOUT ANY WARRANTY; without even the implied warranty of
+MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+GNU Affero General Public License for more details.
+
+You should have received a copy of the GNU Affero General Public License
+along with this program. If not, see .
+*/
+
+package builder
+
+import (
+ corev1 "k8s.io/api/core/v1"
+)
+
+type ServiceAccountBuilder struct {
+ BaseBuilder[corev1.ServiceAccount, *corev1.ServiceAccount, ServiceAccountBuilder]
+}
+
+func NewServiceAccountBuilder(namespace, name string) *ServiceAccountBuilder {
+ builder := &ServiceAccountBuilder{}
+ builder.init(namespace, name, &corev1.ServiceAccount{}, builder)
+ return builder
+}
diff --git a/internal/controller/builder/builder_service_account_test.go b/internal/controller/builder/builder_service_account_test.go
new file mode 100644
index 00000000000..3fc6809644a
--- /dev/null
+++ b/internal/controller/builder/builder_service_account_test.go
@@ -0,0 +1,39 @@
+/*
+Copyright (C) 2022-2023 ApeCloud Co., Ltd
+
+This file is part of KubeBlocks project
+
+This program is free software: you can redistribute it and/or modify
+it under the terms of the GNU Affero General Public License as published by
+the Free Software Foundation, either version 3 of the License, or
+(at your option) any later version.
+
+This program is distributed in the hope that it will be useful
+but WITHOUT ANY WARRANTY; without even the implied warranty of
+MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+GNU Affero General Public License for more details.
+
+You should have received a copy of the GNU Affero General Public License
+along with this program. If not, see .
+*/
+
+package builder
+
+import (
+ . "github.com/onsi/ginkgo/v2"
+ . "github.com/onsi/gomega"
+)
+
+var _ = Describe("service account builder", func() {
+ It("should work well", func() {
+ const (
+ name = "foo"
+ ns = "default"
+ )
+ sa := NewServiceAccountBuilder(ns, name).
+ GetObject()
+
+ Expect(sa.Name).Should(Equal(name))
+ Expect(sa.Namespace).Should(Equal(ns))
+ })
+})
diff --git a/internal/controller/builder/builder_volume_snapshot_class.go b/internal/controller/builder/builder_volume_snapshot_class.go
new file mode 100644
index 00000000000..6224174c60b
--- /dev/null
+++ b/internal/controller/builder/builder_volume_snapshot_class.go
@@ -0,0 +1,44 @@
+/*
+Copyright (C) 2022-2023 ApeCloud Co., Ltd
+
+This file is part of KubeBlocks project
+
+This program is free software: you can redistribute it and/or modify
+it under the terms of the GNU Affero General Public License as published by
+the Free Software Foundation, either version 3 of the License, or
+(at your option) any later version.
+
+This program is distributed in the hope that it will be useful
+but WITHOUT ANY WARRANTY; without even the implied warranty of
+MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+GNU Affero General Public License for more details.
+
+You should have received a copy of the GNU Affero General Public License
+along with this program. If not, see .
+*/
+
+package builder
+
+import (
+ snapshotv1 "github.com/kubernetes-csi/external-snapshotter/client/v6/apis/volumesnapshot/v1"
+)
+
+type VolumeSnapshotClassBuilder struct {
+ BaseBuilder[snapshotv1.VolumeSnapshotClass, *snapshotv1.VolumeSnapshotClass, VolumeSnapshotClassBuilder]
+}
+
+func NewVolumeSnapshotClassBuilder(namespace, name string) *VolumeSnapshotClassBuilder {
+ builder := &VolumeSnapshotClassBuilder{}
+ builder.init(namespace, name, &snapshotv1.VolumeSnapshotClass{}, builder)
+ return builder
+}
+
+func (builder *VolumeSnapshotClassBuilder) SetDriver(driver string) *VolumeSnapshotClassBuilder {
+ builder.get().Driver = driver
+ return builder
+}
+
+func (builder *VolumeSnapshotClassBuilder) SetDeletionPolicy(policy snapshotv1.DeletionPolicy) *VolumeSnapshotClassBuilder {
+ builder.get().DeletionPolicy = policy
+ return builder
+}
diff --git a/internal/controller/builder/builder_volume_snapshot_class_test.go b/internal/controller/builder/builder_volume_snapshot_class_test.go
new file mode 100644
index 00000000000..e00aa618405
--- /dev/null
+++ b/internal/controller/builder/builder_volume_snapshot_class_test.go
@@ -0,0 +1,48 @@
+/*
+Copyright (C) 2022-2023 ApeCloud Co., Ltd
+
+This file is part of KubeBlocks project
+
+This program is free software: you can redistribute it and/or modify
+it under the terms of the GNU Affero General Public License as published by
+the Free Software Foundation, either version 3 of the License, or
+(at your option) any later version.
+
+This program is distributed in the hope that it will be useful
+but WITHOUT ANY WARRANTY; without even the implied warranty of
+MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+GNU Affero General Public License for more details.
+
+You should have received a copy of the GNU Affero General Public License
+along with this program. If not, see .
+*/
+
+package builder
+
+import (
+ . "github.com/onsi/ginkgo/v2"
+ . "github.com/onsi/gomega"
+
+ snapshotv1 "github.com/kubernetes-csi/external-snapshotter/client/v6/apis/volumesnapshot/v1"
+)
+
+var _ = Describe("volume snapshot class builder", func() {
+ It("should work well", func() {
+ const (
+ name = "foo"
+ ns = "default"
+ )
+
+ driver := "openebs-snapshot"
+ policy := snapshotv1.VolumeSnapshotContentRetain
+ vsc := NewVolumeSnapshotClassBuilder(ns, name).
+ SetDriver(driver).
+ SetDeletionPolicy(policy).
+ GetObject()
+
+ Expect(vsc.Name).Should(Equal(name))
+ Expect(vsc.Namespace).Should(Equal(ns))
+ Expect(vsc.Driver).Should(Equal(driver))
+ Expect(vsc.DeletionPolicy).Should(Equal(policy))
+ })
+})
diff --git a/internal/controller/component/affinity_utils.go b/internal/controller/component/affinity_utils.go
index 90a2886695a..0a787a859b9 100644
--- a/internal/controller/component/affinity_utils.go
+++ b/internal/controller/component/affinity_utils.go
@@ -31,7 +31,7 @@ import (
viper "github.com/apecloud/kubeblocks/internal/viperx"
)
-func buildPodTopologySpreadConstraints(
+func BuildPodTopologySpreadConstraints(
cluster *appsv1alpha1.Cluster,
clusterOrCompAffinity *appsv1alpha1.Affinity,
component *SynthesizedComponent,
diff --git a/internal/controller/component/affinity_utils_test.go b/internal/controller/component/affinity_utils_test.go
index ae5a87d5423..08086d270f0 100644
--- a/internal/controller/component/affinity_utils_test.go
+++ b/internal/controller/component/affinity_utils_test.go
@@ -114,7 +114,7 @@ var _ = Describe("affinity utils", func() {
Expect(affinity.PodAntiAffinity.PreferredDuringSchedulingIgnoredDuringExecution).Should(BeEmpty())
Expect(affinity.NodeAffinity.PreferredDuringSchedulingIgnoredDuringExecution).Should(BeEmpty())
- topologySpreadConstraints := buildPodTopologySpreadConstraints(clusterObj, clusterObj.Spec.Affinity, component)
+ topologySpreadConstraints := BuildPodTopologySpreadConstraints(clusterObj, clusterObj.Spec.Affinity, component)
Expect(topologySpreadConstraints[0].WhenUnsatisfiable).Should(Equal(corev1.DoNotSchedule))
Expect(topologySpreadConstraints[0].TopologyKey).Should(Equal(topologyKey))
})
@@ -131,7 +131,7 @@ var _ = Describe("affinity utils", func() {
Expect(affinity.PodAntiAffinity.PreferredDuringSchedulingIgnoredDuringExecution).Should(BeEmpty())
Expect(affinity.NodeAffinity.PreferredDuringSchedulingIgnoredDuringExecution[0].Preference.MatchExpressions[0].Key).Should(Equal(nodeKey))
- topologySpreadConstraints := buildPodTopologySpreadConstraints(clusterObj, clusterObj.Spec.Affinity, component)
+ topologySpreadConstraints := BuildPodTopologySpreadConstraints(clusterObj, clusterObj.Spec.Affinity, component)
Expect(topologySpreadConstraints[0].WhenUnsatisfiable).Should(Equal(corev1.DoNotSchedule))
Expect(topologySpreadConstraints[0].TopologyKey).Should(Equal(topologyKey))
})
@@ -175,7 +175,7 @@ var _ = Describe("affinity utils", func() {
Expect(affinity.PodAntiAffinity.PreferredDuringSchedulingIgnoredDuringExecution[0].Weight).ShouldNot(BeNil())
Expect(affinity.PodAntiAffinity.PreferredDuringSchedulingIgnoredDuringExecution[0].PodAffinityTerm.TopologyKey).Should(Equal(topologyKey))
- topologySpreadConstraints := buildPodTopologySpreadConstraints(clusterObj, clusterObj.Spec.Affinity, component)
+ topologySpreadConstraints := BuildPodTopologySpreadConstraints(clusterObj, clusterObj.Spec.Affinity, component)
Expect(topologySpreadConstraints[0].WhenUnsatisfiable).Should(Equal(corev1.ScheduleAnyway))
Expect(topologySpreadConstraints[0].TopologyKey).Should(Equal(topologyKey))
})
diff --git a/internal/controller/component/component.go b/internal/controller/component/component.go
index 0fdadaea90b..5a7401f4357 100644
--- a/internal/controller/component/component.go
+++ b/internal/controller/component/component.go
@@ -201,6 +201,7 @@ func buildComponent(reqCtx intctrlutil.RequestCtx,
StatefulSpec: clusterCompDefObj.StatefulSpec,
ConsensusSpec: clusterCompDefObj.ConsensusSpec,
ReplicationSpec: clusterCompDefObj.ReplicationSpec,
+ RSMSpec: clusterCompDefObj.RSMSpec,
PodSpec: clusterCompDefObj.PodSpec,
Probes: clusterCompDefObj.Probes,
LogConfigs: clusterCompDefObj.LogConfigs,
@@ -243,7 +244,7 @@ func buildComponent(reqCtx intctrlutil.RequestCtx,
reqCtx.Log.Error(err, "build pod affinity failed.")
return nil, err
}
- component.PodSpec.TopologySpreadConstraints = buildPodTopologySpreadConstraints(cluster, affinity, component)
+ component.PodSpec.TopologySpreadConstraints = BuildPodTopologySpreadConstraints(cluster, affinity, component)
if component.PodSpec.Tolerations, err = BuildTolerations(cluster, clusterCompSpec); err != nil {
reqCtx.Log.Error(err, "build pod tolerations failed.")
return nil, err
@@ -278,18 +279,17 @@ func buildComponent(reqCtx intctrlutil.RequestCtx,
}
}
- // probe container requires a service account with adequate privileges.
- // If probes are required and the serviceAccountName is not set,
+ buildMonitorConfig(clusterCompDefObj, clusterCompSpec, component)
+
+ // lorry container requires a service account with adequate privileges.
+ // If lorry required and the serviceAccountName is not set,
// a default serviceAccountName will be assigned.
if component.ServiceAccountName == "" && component.Probes != nil {
component.ServiceAccountName = "kb-" + component.ClusterName
}
-
// set component.PodSpec.ServiceAccountName
component.PodSpec.ServiceAccountName = component.ServiceAccountName
-
- buildMonitorConfig(clusterCompDefObj, clusterCompSpec, component)
- if err = buildProbeContainers(reqCtx, component); err != nil {
+ if err = buildLorryContainers(reqCtx, component); err != nil {
reqCtx.Log.Error(err, "build probe container failed.")
return nil, err
}
diff --git a/internal/controller/component/component_test.go b/internal/controller/component/component_test.go
index 9fb9b84aff8..b2210e1e667 100644
--- a/internal/controller/component/component_test.go
+++ b/internal/controller/component/component_test.go
@@ -125,7 +125,7 @@ var _ = Describe("component module", func() {
nil,
&clusterVersion.Spec.ComponentVersions[1])
Expect(err).Should(Succeed())
- Expect(len(component.PodSpec.Containers)).Should(Equal(4))
+ Expect(len(component.PodSpec.Containers)).Should(Equal(3))
By("new init container in clusterVersion not in clusterDefinition")
component, err = BuildComponent(
diff --git a/internal/controller/component/cue/probe_template.cue b/internal/controller/component/cue/probe_template.cue
deleted file mode 100644
index 0eae7891988..00000000000
--- a/internal/controller/component/cue/probe_template.cue
+++ /dev/null
@@ -1,53 +0,0 @@
-//Copyright (C) 2022-2023 ApeCloud Co., Ltd
-//
-//This file is part of KubeBlocks project
-//
-//This program is free software: you can redistribute it and/or modify
-//it under the terms of the GNU Affero General Public License as published by
-//the Free Software Foundation, either version 3 of the License, or
-//(at your option) any later version.
-//
-//This program is distributed in the hope that it will be useful
-//but WITHOUT ANY WARRANTY; without even the implied warranty of
-//MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
-//GNU Affero General Public License for more details.
-//
-//You should have received a copy of the GNU Affero General Public License
-//along with this program. If not, see .
-
-probeContainer: {
- image: "registry.cn-hangzhou.aliyuncs.com/google_containers/pause:3.6"
- command: ["/pause"]
- imagePullPolicy: "IfNotPresent"
- name: "string"
- "env": [
- {
- "name": "KB_SERVICE_USER"
- "valueFrom": {
- "secretKeyRef": {
- "key": "username"
- "name": "$(CONN_CREDENTIAL_SECRET_NAME)"
- }
- }
- },
- {
- "name": "KB_SERVICE_PASSWORD"
- "valueFrom": {
- "secretKeyRef": {
- "key": "password"
- "name": "$(CONN_CREDENTIAL_SECRET_NAME)"
- }
- }
- },
- ]
- readinessProbe: {
- exec: {
- command: []
- }
- }
- startupProbe: {
- tcpSocket: {
- port: 3501
- }
- }
-}
diff --git a/internal/controller/component/probe_utils.go b/internal/controller/component/probe_utils.go
index f23fe2cd3ce..e9e6b2a2c6c 100644
--- a/internal/controller/component/probe_utils.go
+++ b/internal/controller/component/probe_utils.go
@@ -20,26 +20,22 @@ along with this program. If not, see .
package component
import (
- "embed"
"encoding/json"
"fmt"
"strconv"
- "strings"
- "github.com/leaanthony/debme"
corev1 "k8s.io/api/core/v1"
"k8s.io/apimachinery/pkg/util/intstr"
appsv1alpha1 "github.com/apecloud/kubeblocks/apis/apps/v1alpha1"
"github.com/apecloud/kubeblocks/internal/constant"
+ "github.com/apecloud/kubeblocks/internal/controller/builder"
intctrlutil "github.com/apecloud/kubeblocks/internal/controllerutil"
viper "github.com/apecloud/kubeblocks/internal/viperx"
)
const (
// http://localhost:/v1.0/bindings/
- checkRoleURIFormat = "/v1.0/bindings/%s?operation=checkRole&workloadType=%s"
- getGlobalInfoFormat = "/v1.0/bindings/%s?operation=getGlobalInfo"
checkRunningURIFormat = "/v1.0/bindings/%s?operation=checkRunning"
checkStatusURIFormat = "/v1.0/bindings/%s?operation=checkStatus"
volumeProtectionURIFormat = "/v1.0/bindings/%s?operation=volumeProtection"
@@ -48,9 +44,6 @@ const (
)
var (
- //go:embed cue/*
- cueTemplates embed.FS
-
// default probe setting for volume protection.
defaultVolumeProtectionProbe = appsv1alpha1.ClusterDefinitionProbe{
PeriodSeconds: 60,
@@ -59,87 +52,98 @@ var (
}
)
-func buildProbeContainers(reqCtx intctrlutil.RequestCtx, component *SynthesizedComponent) error {
- container, err := buildProbeContainer()
- if err != nil {
- return err
- }
-
- probeContainers := []corev1.Container{}
- componentProbes := component.Probes
- if componentProbes == nil {
+func buildLorryContainers(reqCtx intctrlutil.RequestCtx, component *SynthesizedComponent) error {
+ container := buildBasicContainer()
+ var lorryContainers []corev1.Container
+ componentLorry := component.Probes
+ if componentLorry == nil {
return nil
}
- reqCtx.Log.V(3).Info("probe", "settings", componentProbes)
- probeSvcHTTPPort := viper.GetInt32("PROBE_SERVICE_HTTP_PORT")
- probeSvcGRPCPort := viper.GetInt32("PROBE_SERVICE_GRPC_PORT")
- availablePorts, err := getAvailableContainerPorts(component.PodSpec.Containers, []int32{probeSvcHTTPPort, probeSvcGRPCPort})
- probeSvcHTTPPort = availablePorts[0]
- probeSvcGRPCPort = availablePorts[1]
+ reqCtx.Log.V(3).Info("lorry", "settings", componentLorry)
+ lorrySvcHTTPPort := viper.GetInt32("PROBE_SERVICE_HTTP_PORT")
+ lorrySvcGRPCPort := viper.GetInt32("PROBE_SERVICE_GRPC_PORT")
+ // override by new env name
+ if viper.IsSet("LORRY_SERVICE_HTTP_PORT") {
+ lorrySvcHTTPPort = viper.GetInt32("LORRY_SERVICE_HTTP_PORT")
+ }
+ if viper.IsSet("LORRY_SERVICE_GRPC_PORT") {
+ lorrySvcGRPCPort = viper.GetInt32("LORRY_SERVICE_GRPC_PORT")
+ }
+ availablePorts, err := getAvailableContainerPorts(component.PodSpec.Containers, []int32{lorrySvcHTTPPort, lorrySvcGRPCPort})
+ lorrySvcHTTPPort = availablePorts[0]
+ lorrySvcGRPCPort = availablePorts[1]
if err != nil {
- reqCtx.Log.Info("get probe container port failed", "error", err)
+ reqCtx.Log.Info("get lorry container port failed", "error", err)
return err
}
- // injectHttp2Shell(component.PodSpec)
-
- if componentProbes.RoleProbe != nil {
- roleChangedContainer := container.DeepCopy()
- buildRoleProbeContainer(component, roleChangedContainer, componentProbes.RoleProbe, int(probeSvcHTTPPort), component.PodSpec)
- probeContainers = append(probeContainers, *roleChangedContainer)
- }
-
- if componentProbes.StatusProbe != nil {
+ if componentLorry.StatusProbe != nil {
statusProbeContainer := container.DeepCopy()
- buildStatusProbeContainer(component.CharacterType, statusProbeContainer, componentProbes.StatusProbe, int(probeSvcHTTPPort))
- probeContainers = append(probeContainers, *statusProbeContainer)
+ buildStatusProbeContainer(component.CharacterType, statusProbeContainer, componentLorry.StatusProbe, int(lorrySvcHTTPPort))
+ lorryContainers = append(lorryContainers, *statusProbeContainer)
}
- if componentProbes.RunningProbe != nil {
+ if componentLorry.RunningProbe != nil {
runningProbeContainer := container.DeepCopy()
- buildRunningProbeContainer(component.CharacterType, runningProbeContainer, componentProbes.RunningProbe, int(probeSvcHTTPPort))
- probeContainers = append(probeContainers, *runningProbeContainer)
+ buildRunningProbeContainer(component.CharacterType, runningProbeContainer, componentLorry.RunningProbe, int(lorrySvcHTTPPort))
+ lorryContainers = append(lorryContainers, *runningProbeContainer)
}
if volumeProtectionEnabled(component) {
c := container.DeepCopy()
- buildVolumeProtectionProbeContainer(component.CharacterType, c, int(probeSvcHTTPPort))
- probeContainers = append(probeContainers, *c)
+ buildVolumeProtectionProbeContainer(component.CharacterType, c, int(lorrySvcHTTPPort))
+ lorryContainers = append(lorryContainers, *c)
}
- if len(probeContainers) >= 1 {
- container := &probeContainers[0]
- buildProbeServiceContainer(component, container, int(probeSvcHTTPPort), int(probeSvcGRPCPort))
+ // inject WeSyncer(currently part of Lorry) in cluster controller.
+ // as all the above features share the lorry service, only one lorry need to be injected.
+ // if none of the above feature enabled, WeSyncer still need to be injected for the HA feature functions well.
+ if len(lorryContainers) == 0 {
+ weSyncerContainer := container.DeepCopy()
+ buildWeSyncerContainer(weSyncerContainer, int(lorrySvcHTTPPort))
+ lorryContainers = append(lorryContainers, *weSyncerContainer)
}
- reqCtx.Log.V(1).Info("probe", "containers", probeContainers)
- component.PodSpec.Containers = append(component.PodSpec.Containers, probeContainers...)
+ buildLorryServiceContainer(component, &lorryContainers[0], int(lorrySvcHTTPPort), int(lorrySvcGRPCPort))
+
+ reqCtx.Log.V(1).Info("lorry", "containers", lorryContainers)
+ component.PodSpec.Containers = append(component.PodSpec.Containers, lorryContainers...)
return nil
}
-func buildProbeContainer() (*corev1.Container, error) {
- cueFS, _ := debme.FS(cueTemplates, "cue")
-
- cueTpl, err := intctrlutil.NewCUETplFromBytes(cueFS.ReadFile("probe_template.cue"))
- if err != nil {
- return nil, err
- }
- cueValue := intctrlutil.NewCUEBuilder(*cueTpl)
- probeContainerByte, err := cueValue.Lookup("probeContainer")
- if err != nil {
- return nil, err
- }
- container := &corev1.Container{}
- if err = json.Unmarshal(probeContainerByte, container); err != nil {
- return nil, err
- }
- return container, nil
+func buildBasicContainer() *corev1.Container {
+ return builder.NewContainerBuilder("string").
+ SetImage("registry.cn-hangzhou.aliyuncs.com/google_containers/pause:3.6").
+ SetImagePullPolicy(corev1.PullIfNotPresent).
+ AddCommands("/pause").
+ AddEnv(corev1.EnvVar{
+ Name: "KB_SERVICE_USER",
+ ValueFrom: &corev1.EnvVarSource{
+ SecretKeyRef: &corev1.SecretKeySelector{
+ Key: "username",
+ LocalObjectReference: corev1.LocalObjectReference{Name: "$(CONN_CREDENTIAL_SECRET_NAME)"},
+ },
+ }},
+ corev1.EnvVar{
+ Name: "KB_SERVICE_PASSWORD",
+ ValueFrom: &corev1.EnvVarSource{
+ SecretKeyRef: &corev1.SecretKeySelector{
+ Key: "password",
+ LocalObjectReference: corev1.LocalObjectReference{Name: "$(CONN_CREDENTIAL_SECRET_NAME)"},
+ },
+ },
+ }).
+ SetStartupProbe(corev1.Probe{
+ ProbeHandler: corev1.ProbeHandler{
+ TCPSocket: &corev1.TCPSocketAction{Port: intstr.FromInt(3501)},
+ }}).
+ GetObject()
}
-func buildProbeServiceContainer(component *SynthesizedComponent, container *corev1.Container, probeSvcHTTPPort int, probeSvcGRPCPort int) {
+func buildLorryServiceContainer(component *SynthesizedComponent, container *corev1.Container, probeSvcHTTPPort int, probeSvcGRPCPort int) {
container.Image = viper.GetString(constant.KBToolsImage)
container.ImagePullPolicy = corev1.PullPolicy(viper.GetString(constant.KBImagePullPolicy))
- container.Command = []string{"probe",
+ container.Command = []string{"lorry",
"--port", strconv.Itoa(probeSvcHTTPPort)}
if len(component.PodSpec.Containers) > 0 {
@@ -174,14 +178,6 @@ func buildProbeServiceContainer(component *SynthesizedComponent, container *core
}
}
- roles := getComponentRoles(component)
- rolesJSON, _ := json.Marshal(roles)
- container.Env = append(container.Env, corev1.EnvVar{
- Name: constant.KBEnvServiceRoles,
- Value: string(rolesJSON),
- ValueFrom: nil,
- })
-
container.Env = append(container.Env, corev1.EnvVar{
Name: constant.KBEnvCharacterType,
Value: component.CharacterType,
@@ -194,12 +190,6 @@ func buildProbeServiceContainer(component *SynthesizedComponent, container *core
ValueFrom: nil,
})
- container.Env = append(container.Env, corev1.EnvVar{
- Name: "KB_RSM_ACTION_SVC_LIST",
- Value: viper.GetString("KB_RSM_ACTION_SVC_LIST"),
- ValueFrom: nil,
- })
-
container.Ports = []corev1.ContainerPort{
{
ContainerPort: int32(probeSvcHTTPPort),
@@ -212,93 +202,44 @@ func buildProbeServiceContainer(component *SynthesizedComponent, container *core
Protocol: "TCP",
}}
- // pass the volume protection spec to probe container through env.
+ // pass the volume protection spec to lorry container through env.
if volumeProtectionEnabled(component) {
container.Env = append(container.Env, env4VolumeProtection(*component.VolumeProtection))
}
}
-func getComponentRoles(component *SynthesizedComponent) map[string]string {
- var roles = map[string]string{}
- if component.ConsensusSpec == nil {
- return roles
- }
-
- consensus := component.ConsensusSpec
- roles[strings.ToLower(consensus.Leader.Name)] = string(consensus.Leader.AccessMode)
- for _, follower := range consensus.Followers {
- roles[strings.ToLower(follower.Name)] = string(follower.AccessMode)
- }
- if consensus.Learner != nil {
- roles[strings.ToLower(consensus.Learner.Name)] = string(consensus.Learner.AccessMode)
- }
- return roles
-}
-
-func buildRoleProbeContainer(component *SynthesizedComponent, roleChangedContainer *corev1.Container,
- probeSetting *appsv1alpha1.ClusterDefinitionProbe, probeSvcHTTPPort int, pod *corev1.PodSpec) {
- roleChangedContainer.Name = constant.RoleProbeContainerName
- probe := roleChangedContainer.ReadinessProbe
- bindingType := strings.ToLower(component.CharacterType)
- workloadType := component.WorkloadType
- httpGet := &corev1.HTTPGetAction{}
- httpGet.Path = fmt.Sprintf(checkRoleURIFormat, bindingType, workloadType)
- httpGet.Port = intstr.FromInt(probeSvcHTTPPort)
- probe.Exec = nil
- probe.HTTPGet = httpGet
- probe.PeriodSeconds = probeSetting.PeriodSeconds
- probe.TimeoutSeconds = probeSetting.TimeoutSeconds
- probe.FailureThreshold = probeSetting.FailureThreshold
- roleChangedContainer.StartupProbe.TCPSocket.Port = intstr.FromInt(probeSvcHTTPPort)
-
- // -> uncomment it to enable snapshot to cluster
-
- // base := probeSvcHTTPPort + 2
- // portNeeded := len(probeSetting.Actions)
- // activePorts := make([]int32, portNeeded)
- // for i := 0; i < portNeeded; i++ {
- // activePorts[i] = int32(base + i)
- // }
- // activePorts, err := getAvailableContainerPorts(pod.Containers, activePorts)
- // if err != nil {
- // return
- // }
- // marshal, err := json.Marshal(activePorts)
- // if err != nil {
- // return
- // }
- // viper.Set("KB_RSM_ACTION_SVC_LIST", string(marshal))
-
- // injectProbeUtilImages(pod, probeSetting, activePorts, "/role", "checkrole", roleChangedContainer.Env)
+func buildWeSyncerContainer(weSyncerContainer *corev1.Container, probeSvcHTTPPort int) {
+ weSyncerContainer.Name = constant.WeSyncerContainerName
+ weSyncerContainer.StartupProbe.TCPSocket.Port = intstr.FromInt(probeSvcHTTPPort)
}
func buildStatusProbeContainer(characterType string, statusProbeContainer *corev1.Container,
probeSetting *appsv1alpha1.ClusterDefinitionProbe, probeSvcHTTPPort int) {
statusProbeContainer.Name = constant.StatusProbeContainerName
- probe := statusProbeContainer.ReadinessProbe
+ probe := &corev1.Probe{}
httpGet := &corev1.HTTPGetAction{}
httpGet.Path = fmt.Sprintf(checkStatusURIFormat, characterType)
httpGet.Port = intstr.FromInt(probeSvcHTTPPort)
- probe.Exec = nil
probe.HTTPGet = httpGet
probe.PeriodSeconds = probeSetting.PeriodSeconds
probe.TimeoutSeconds = probeSetting.TimeoutSeconds
probe.FailureThreshold = probeSetting.FailureThreshold
+ statusProbeContainer.ReadinessProbe = probe
statusProbeContainer.StartupProbe.TCPSocket.Port = intstr.FromInt(probeSvcHTTPPort)
}
func buildRunningProbeContainer(characterType string, runningProbeContainer *corev1.Container,
probeSetting *appsv1alpha1.ClusterDefinitionProbe, probeSvcHTTPPort int) {
runningProbeContainer.Name = constant.RunningProbeContainerName
- probe := runningProbeContainer.ReadinessProbe
+ probe := &corev1.Probe{}
httpGet := &corev1.HTTPGetAction{}
httpGet.Path = fmt.Sprintf(checkRunningURIFormat, characterType)
httpGet.Port = intstr.FromInt(probeSvcHTTPPort)
- probe.Exec = nil
probe.HTTPGet = httpGet
probe.PeriodSeconds = probeSetting.PeriodSeconds
probe.TimeoutSeconds = probeSetting.TimeoutSeconds
probe.FailureThreshold = probeSetting.FailureThreshold
+ runningProbeContainer.ReadinessProbe = probe
runningProbeContainer.StartupProbe.TCPSocket.Port = intstr.FromInt(probeSvcHTTPPort)
}
@@ -308,15 +249,15 @@ func volumeProtectionEnabled(component *SynthesizedComponent) bool {
func buildVolumeProtectionProbeContainer(characterType string, c *corev1.Container, probeSvcHTTPPort int) {
c.Name = constant.VolumeProtectionProbeContainerName
- probe := c.ReadinessProbe
+ probe := &corev1.Probe{}
httpGet := &corev1.HTTPGetAction{}
httpGet.Path = fmt.Sprintf(volumeProtectionURIFormat, characterType)
httpGet.Port = intstr.FromInt(probeSvcHTTPPort)
- probe.Exec = nil
probe.HTTPGet = httpGet
probe.PeriodSeconds = defaultVolumeProtectionProbe.PeriodSeconds
probe.TimeoutSeconds = defaultVolumeProtectionProbe.TimeoutSeconds
probe.FailureThreshold = defaultVolumeProtectionProbe.FailureThreshold
+ c.ReadinessProbe = probe
c.StartupProbe.TCPSocket.Port = intstr.FromInt(probeSvcHTTPPort)
}
@@ -330,72 +271,3 @@ func env4VolumeProtection(spec appsv1alpha1.VolumeProtectionSpec) corev1.EnvVar
Value: string(value),
}
}
-
-// func injectHttp2Shell(pod *corev1.PodSpec) {
-// // inject shared volume
-// agentVolume := corev1.Volume{
-// Name: constant.ProbeAgentMountName,
-// VolumeSource: corev1.VolumeSource{
-// EmptyDir: &corev1.EmptyDirVolumeSource{},
-// },
-// }
-// pod.Volumes = append(pod.Volumes, agentVolume)
-//
-// // inject shell2http
-// volumeMount := corev1.VolumeMount{
-// Name: constant.ProbeAgentMountName,
-// MountPath: constant.ProbeAgentMountPath,
-// }
-// binPath := strings.Join([]string{constant.ProbeAgentMountPath, constant.ProbeAgent}, "/")
-// initContainer := corev1.Container{
-// Name: constant.ProbeAgent,
-// Image: constant.ProbeAgentImage,
-// ImagePullPolicy: corev1.PullIfNotPresent,
-// VolumeMounts: []corev1.VolumeMount{volumeMount},
-// Command: []string{
-// "cp",
-// constant.OriginBinaryPath,
-// binPath,
-// },
-// }
-// pod.InitContainers = append(pod.InitContainers, initContainer)
-//}
-//
-// func injectProbeUtilImages(pod *corev1.PodSpec, probeSetting *appsv1alpha1.ClusterDefinitionProbe,
-// port []int32, path, usage string,
-// credentialEnv []corev1.EnvVar) {
-// // todo: uncomment to enable new lorry way
-// // actions := probeSetting.Actions
-// // volumeMount := corev1.VolumeMount{
-// // Name: constant.ProbeAgentMountName,
-// // MountPath: constant.ProbeAgentMountPath,
-// // }
-// // binPath := strings.Join([]string{constant.ProbeAgentMountPath, constant.ProbeAgent}, "/")
-// //
-// // for i, action := range actions {
-// // image := action.Image
-// // if len(action.Image) == 0 {
-// // image = constant.DefaultActionImage
-// // }
-// //
-// // command := []string{
-// // binPath,
-// // "-port", fmt.Sprintf("%d", port[i]),
-// // "-export-all-vars",
-// // "-form",
-// // path,
-// // strings.Join(action.Command, " "),
-// // }
-// //
-// // container := corev1.Container{
-// // Name: fmt.Sprintf("%s-action-%d", usage, i),
-// // Image: image,
-// // ImagePullPolicy: corev1.PullIfNotPresent,
-// // VolumeMounts: []corev1.VolumeMount{volumeMount},
-// // Env: credentialEnv,
-// // Command: command,
-// // }
-// //
-// // pod.Containers = append(pod.Containers, container)
-// // }
-// }
diff --git a/internal/controller/component/probe_utils_test.go b/internal/controller/component/probe_utils_test.go
index 737083bf40e..ed022cd874b 100644
--- a/internal/controller/component/probe_utils_test.go
+++ b/internal/controller/component/probe_utils_test.go
@@ -44,9 +44,7 @@ var _ = Describe("probe_utils", func() {
var clusterDefProbe *appsv1alpha1.ClusterDefinitionProbe
BeforeEach(func() {
- var err error
- container, err = buildProbeContainer()
- Expect(err).NotTo(HaveOccurred())
+ container = buildBasicContainer()
probeServiceHTTPPort, probeServiceGrpcPort = 3501, 50001
clusterDefProbe = &appsv1alpha1.ClusterDefinitionProbe{}
@@ -95,20 +93,13 @@ var _ = Describe("probe_utils", func() {
Ctx: ctx,
Log: logger,
}
- Expect(buildProbeContainers(reqCtx, component)).Should(Succeed())
- Expect(len(component.PodSpec.Containers)).Should(Equal(3))
+ Expect(buildLorryContainers(reqCtx, component)).Should(Succeed())
+ Expect(len(component.PodSpec.Containers)).Should(Equal(2))
Expect(component.PodSpec.Containers[0].Command).ShouldNot(BeEmpty())
})
- It("should build role changed probe container", func() {
- synthesizedComponent := &SynthesizedComponent{CharacterType: "wesql"}
- pod := &corev1.PodSpec{}
- buildRoleProbeContainer(synthesizedComponent, container, clusterDefProbe, probeServiceHTTPPort, pod)
- Expect(container.ReadinessProbe.HTTPGet).ShouldNot(BeNil())
- })
-
It("should build role service container", func() {
- buildProbeServiceContainer(component, container, probeServiceHTTPPort, probeServiceGrpcPort)
+ buildLorryServiceContainer(component, container, probeServiceHTTPPort, probeServiceGrpcPort)
Expect(container.Command).ShouldNot(BeEmpty())
})
@@ -140,8 +131,8 @@ var _ = Describe("probe_utils", func() {
},
},
}
- Expect(buildProbeContainers(reqCtx, component)).Should(Succeed())
- Expect(len(component.PodSpec.Containers)).Should(Equal(4))
+ Expect(buildLorryContainers(reqCtx, component)).Should(Succeed())
+ Expect(len(component.PodSpec.Containers)).Should(Equal(3))
})
It("build volume protection probe container with RBAC", func() {
@@ -163,8 +154,8 @@ var _ = Describe("probe_utils", func() {
},
}
viper.SetDefault(constant.EnableRBACManager, true)
- Expect(buildProbeContainers(reqCtx, component)).Should(Succeed())
- Expect(len(component.PodSpec.Containers)).Should(Equal(4))
+ Expect(buildLorryContainers(reqCtx, component)).Should(Succeed())
+ Expect(len(component.PodSpec.Containers)).Should(Equal(3))
spec := &appsv1alpha1.VolumeProtectionSpec{}
for _, e := range component.PodSpec.Containers[0].Env {
if e.Name == constant.KBEnvVolumeProtectionSpec {
diff --git a/internal/controller/component/suite_test.go b/internal/controller/component/suite_test.go
index 4e4d91fff33..071691f53c7 100644
--- a/internal/controller/component/suite_test.go
+++ b/internal/controller/component/suite_test.go
@@ -39,7 +39,7 @@ import (
"sigs.k8s.io/controller-runtime/pkg/log/zap"
appsv1alpha1 "github.com/apecloud/kubeblocks/apis/apps/v1alpha1"
- dataprotectionv1alpha1 "github.com/apecloud/kubeblocks/apis/dataprotection/v1alpha1"
+ dpv1alpha1 "github.com/apecloud/kubeblocks/apis/dataprotection/v1alpha1"
intctrlutil "github.com/apecloud/kubeblocks/internal/controllerutil"
"github.com/apecloud/kubeblocks/internal/testutil"
viper "github.com/apecloud/kubeblocks/internal/viperx"
@@ -99,7 +99,7 @@ var _ = BeforeSuite(func() {
err = appsv1alpha1.AddToScheme(scheme.Scheme)
Expect(err).NotTo(HaveOccurred())
- err = dataprotectionv1alpha1.AddToScheme(scheme.Scheme)
+ err = dpv1alpha1.AddToScheme(scheme.Scheme)
Expect(err).NotTo(HaveOccurred())
err = snapshotv1.AddToScheme(scheme.Scheme)
diff --git a/internal/controller/component/type.go b/internal/controller/component/type.go
index d5691697c72..58dd1a9d713 100644
--- a/internal/controller/component/type.go
+++ b/internal/controller/component/type.go
@@ -47,6 +47,7 @@ type SynthesizedComponent struct {
StatefulSpec *v1alpha1.StatefulSetSpec `json:"statefulSpec,omitempty"`
ConsensusSpec *v1alpha1.ConsensusSetSpec `json:"consensusSpec,omitempty"`
ReplicationSpec *v1alpha1.ReplicationSetSpec `json:"replicationSpec,omitempty"`
+ RSMSpec *v1alpha1.RSMSpec `json:"rsmSpec,omitempty"`
PodSpec *corev1.PodSpec `json:"podSpec,omitempty"`
Services []corev1.Service `json:"services,omitempty"`
Probes *v1alpha1.ClusterDefinitionProbes `json:"probes,omitempty"`
diff --git a/internal/controller/configuration/configuration_test.go b/internal/controller/configuration/configuration_test.go
index 62c1c3d5d78..637c4003198 100644
--- a/internal/controller/configuration/configuration_test.go
+++ b/internal/controller/configuration/configuration_test.go
@@ -21,14 +21,15 @@ package configuration
import (
. "github.com/onsi/gomega"
+
+ corev1 "k8s.io/api/core/v1"
"k8s.io/apimachinery/pkg/types"
+ "sigs.k8s.io/controller-runtime/pkg/client"
appsv1alpha1 "github.com/apecloud/kubeblocks/apis/apps/v1alpha1"
"github.com/apecloud/kubeblocks/internal/controller/component"
intctrlutil "github.com/apecloud/kubeblocks/internal/controllerutil"
testapps "github.com/apecloud/kubeblocks/internal/testutil/apps"
- corev1 "k8s.io/api/core/v1"
- "sigs.k8s.io/controller-runtime/pkg/client"
)
const clusterDefName = "test-clusterdef"
diff --git a/internal/controller/configuration/operator_test.go b/internal/controller/configuration/operator_test.go
index 23027af26eb..7db807a94a0 100644
--- a/internal/controller/configuration/operator_test.go
+++ b/internal/controller/configuration/operator_test.go
@@ -53,8 +53,7 @@ var _ = Describe("ConfigurationOperatorTest", func() {
var k8sMockClient *testutil.K8sClientMockHelper
mockStatefulSet := func() *appsv1.StatefulSet {
- envConfig, err := factory.BuildEnvConfig(clusterObj, clusterComponent)
- Expect(err).Should(Succeed())
+ envConfig := factory.BuildEnvConfig(clusterObj, clusterComponent)
stsObj, err := factory.BuildSts(intctrlutil.RequestCtx{
Ctx: ctx,
Log: logger,
@@ -93,8 +92,6 @@ var _ = Describe("ConfigurationOperatorTest", func() {
cfgcore.GenerateComponentConfigurationName(clusterName, mysqlCompName)).
ClusterRef(clusterName).
Component(mysqlCompName).
- ClusterVerRef(clusterVersionName).
- ClusterDefRef(clusterDefName).
GetObject()
configConstraint = &appsv1alpha1.ConfigConstraint{
ObjectMeta: metav1.ObjectMeta{
diff --git a/internal/controller/configuration/pipeline.go b/internal/controller/configuration/pipeline.go
index c1dc0af389a..cdc159c093b 100644
--- a/internal/controller/configuration/pipeline.go
+++ b/internal/controller/configuration/pipeline.go
@@ -20,8 +20,6 @@ along with this program. If not, see .
package configuration
import (
- "encoding/json"
- "reflect"
"strconv"
corev1 "k8s.io/api/core/v1"
@@ -145,16 +143,17 @@ func (p *pipeline) UpdateConfigurationStatus() *pipeline {
}
existing := p.ConfigurationObj
+ reversion := fromConfiguration(existing)
patch := client.MergeFrom(existing)
updated := existing.DeepCopy()
for _, item := range existing.Spec.ConfigItemDetails {
- checkAndUpdateItemStatus(updated, item)
+ checkAndUpdateItemStatus(updated, item, reversion)
}
return p.ResourceFetcher.Client.Status().Patch(p.Context, updated, patch)
})
}
-func checkAndUpdateItemStatus(updated *appsv1alpha1.Configuration, item appsv1alpha1.ConfigurationItemDetail) {
+func checkAndUpdateItemStatus(updated *appsv1alpha1.Configuration, item appsv1alpha1.ConfigurationItemDetail, reversion string) {
foundStatus := func(name string) *appsv1alpha1.ConfigurationItemDetailStatus {
for i := range updated.Status.ConfigurationItemStatus {
status := &updated.Status.ConfigurationItemStatus[i]
@@ -172,8 +171,9 @@ func checkAndUpdateItemStatus(updated *appsv1alpha1.Configuration, item appsv1al
if status == nil {
updated.Status.ConfigurationItemStatus = append(updated.Status.ConfigurationItemStatus,
appsv1alpha1.ConfigurationItemDetailStatus{
- Name: item.Name,
- Phase: appsv1alpha1.CInitPhase,
+ Name: item.Name,
+ Phase: appsv1alpha1.CInitPhase,
+ UpdateRevision: reversion,
})
}
}
@@ -208,14 +208,11 @@ func (p *pipeline) createConfiguration() *appsv1alpha1.Configuration {
builder := builder.NewConfigurationBuilder(p.Namespace,
core.GenerateComponentConfigurationName(p.ClusterName, p.ComponentName),
)
-
for _, template := range p.ctx.Component.ConfigTemplates {
- builder.AddConfigurationItem(template.Name)
+ builder.AddConfigurationItem(template)
}
return builder.Component(p.ComponentName).
ClusterRef(p.ClusterName).
- ClusterDefRef(p.ctx.Cluster.Spec.ClusterDefRef).
- ClusterVerRef(p.ctx.Cluster.Spec.ClusterVersionRef).
GetObject()
}
@@ -258,7 +255,7 @@ func (p *updatePipeline) isDone() bool {
func (p *updatePipeline) PrepareForTemplate() *updatePipeline {
buildTemplate := func() (err error) {
- p.reconcile = !IsApplyConfigChanged(p.ConfigMapObj, p.item)
+ p.reconcile = !intctrlutil.IsApplyConfigChanged(p.ConfigMapObj, p.item)
if p.isDone() {
return
}
@@ -273,23 +270,6 @@ func (p *updatePipeline) PrepareForTemplate() *updatePipeline {
return p.Wrap(buildTemplate)
}
-func IsApplyConfigChanged(cm *corev1.ConfigMap, item appsv1alpha1.ConfigurationItemDetail) bool {
- if cm == nil {
- return false
- }
-
- lastAppliedVersion, ok := cm.Annotations[constant.ConfigAppliedVersionAnnotationKey]
- if !ok {
- return false
- }
- var target appsv1alpha1.ConfigurationItemDetail
- if err := json.Unmarshal([]byte(lastAppliedVersion), &target); err != nil {
- return false
- }
-
- return reflect.DeepEqual(target, item)
-}
-
func (p *updatePipeline) ConfigSpec() *appsv1alpha1.ComponentConfigSpec {
return p.configSpec
}
@@ -311,7 +291,7 @@ func (p *updatePipeline) RerenderTemplate() *updatePipeline {
if p.isDone() {
return
}
- if needRerender(p.ConfigMapObj, p.item) {
+ if intctrlutil.IsRerender(p.ConfigMapObj, p.item) {
p.newCM, err = p.renderWrapper.rerenderConfigTemplate(p.ctx.Cluster, p.ctx.Component, *p.configSpec, &p.item)
} else {
p.newCM = p.ConfigMapObj.DeepCopy()
@@ -359,32 +339,28 @@ func (p *updatePipeline) UpdateConfigVersion(revision string) *updatePipeline {
if p.isDone() {
return nil
}
+
+ if err := updateConfigMetaForCM(p.newCM, &p.item, revision); err != nil {
+ return err
+ }
annotations := p.newCM.Annotations
if annotations == nil {
annotations = make(map[string]string)
}
- b, err := json.Marshal(p.item)
- if err != nil {
- return err
- }
- annotations[constant.ConfigAppliedVersionAnnotationKey] = string(b)
- hash, _ := cfgutil.ComputeHash(p.newCM.Data)
- annotations[constant.CMInsCurrentConfigurationHashLabelKey] = hash
- annotations[constant.ConfigurationRevision] = revision
- annotations[constant.CMConfigurationTemplateVersion] = p.item.Version
+
// delete disable reconcile annotation
if _, ok := annotations[constant.DisableUpgradeInsConfigurationAnnotationKey]; ok {
annotations[constant.DisableUpgradeInsConfigurationAnnotationKey] = strconv.FormatBool(false)
}
p.newCM.Annotations = annotations
- p.itemStatus.UpdateRevision = revision
+ // p.itemStatus.UpdateRevision = revision
return nil
})
}
func (p *updatePipeline) Sync() *updatePipeline {
return p.Wrap(func() error {
- if p.ConfigConstraintObj != nil {
+ if p.ConfigConstraintObj != nil && !p.isDone() {
if err := SyncEnvConfigmap(*p.configSpec, p.newCM, &p.ConfigConstraintObj.Spec, p.Client, p.Context); err != nil {
return err
}
@@ -414,18 +390,3 @@ func (p *updatePipeline) SyncStatus() *updatePipeline {
return
})
}
-
-func needRerender(obj *corev1.ConfigMap, item appsv1alpha1.ConfigurationItemDetail) bool {
- if obj == nil {
- return true
- }
- if item.Version == "" {
- return false
- }
-
- version, ok := obj.Annotations[constant.CMConfigurationTemplateVersion]
- if !ok || version != item.Version {
- return true
- }
- return false
-}
diff --git a/internal/controller/configuration/pipeline_test.go b/internal/controller/configuration/pipeline_test.go
index b24974ffa1e..89d935269ab 100644
--- a/internal/controller/configuration/pipeline_test.go
+++ b/internal/controller/configuration/pipeline_test.go
@@ -57,8 +57,7 @@ var _ = Describe("ConfigurationPipelineTest", func() {
var k8sMockClient *testutil.K8sClientMockHelper
mockStatefulSet := func() *appsv1.StatefulSet {
- envConfig, err := factory.BuildEnvConfig(clusterObj, clusterComponent)
- Expect(err).Should(Succeed())
+ envConfig := factory.BuildEnvConfig(clusterObj, clusterComponent)
stsObj, err := factory.BuildSts(intctrlutil.RequestCtx{
Ctx: ctx,
Log: logger,
@@ -125,8 +124,6 @@ max_connections = '1000'
cfgcore.GenerateComponentConfigurationName(clusterName, mysqlCompName)).
ClusterRef(clusterName).
Component(mysqlCompName).
- ClusterVerRef(clusterVersionName).
- ClusterDefRef(clusterDefName).
GetObject()
configConstraint = &appsv1alpha1.ConfigConstraint{
ObjectMeta: metav1.ObjectMeta{
diff --git a/internal/controller/configuration/suite_test.go b/internal/controller/configuration/suite_test.go
index 236a37cf5cf..42d876b8f5d 100644
--- a/internal/controller/configuration/suite_test.go
+++ b/internal/controller/configuration/suite_test.go
@@ -39,7 +39,7 @@ import (
"sigs.k8s.io/controller-runtime/pkg/log/zap"
appsv1alpha1 "github.com/apecloud/kubeblocks/apis/apps/v1alpha1"
- dataprotectionv1alpha1 "github.com/apecloud/kubeblocks/apis/dataprotection/v1alpha1"
+ dpv1alpha1 "github.com/apecloud/kubeblocks/apis/dataprotection/v1alpha1"
intctrlutil "github.com/apecloud/kubeblocks/internal/controllerutil"
"github.com/apecloud/kubeblocks/internal/testutil"
viper "github.com/apecloud/kubeblocks/internal/viperx"
@@ -100,7 +100,7 @@ var _ = BeforeSuite(func() {
err = appsv1alpha1.AddToScheme(scheme.Scheme)
Expect(err).NotTo(HaveOccurred())
- err = dataprotectionv1alpha1.AddToScheme(scheme.Scheme)
+ err = dpv1alpha1.AddToScheme(scheme.Scheme)
Expect(err).NotTo(HaveOccurred())
err = snapshotv1.AddToScheme(scheme.Scheme)
diff --git a/internal/controller/configuration/template_wrapper.go b/internal/controller/configuration/template_wrapper.go
index 8f82e94761b..e774224c992 100644
--- a/internal/controller/configuration/template_wrapper.go
+++ b/internal/controller/configuration/template_wrapper.go
@@ -23,6 +23,7 @@ import (
"context"
"encoding/json"
"reflect"
+ "strconv"
"strings"
corev1 "k8s.io/api/core/v1"
@@ -34,6 +35,7 @@ import (
appsv1alpha1 "github.com/apecloud/kubeblocks/apis/apps/v1alpha1"
"github.com/apecloud/kubeblocks/internal/configuration/core"
+ cfgutil "github.com/apecloud/kubeblocks/internal/configuration/util"
"github.com/apecloud/kubeblocks/internal/configuration/validate"
"github.com/apecloud/kubeblocks/internal/constant"
"github.com/apecloud/kubeblocks/internal/controller/component"
@@ -97,6 +99,7 @@ func (wrapper *renderWrapper) checkRerenderTemplateSpec(cfgCMName string, localO
func (wrapper *renderWrapper) renderConfigTemplate(cluster *appsv1alpha1.Cluster,
component *component.SynthesizedComponent, localObjs []client.Object, configuration *appsv1alpha1.Configuration) error {
scheme, _ := appsv1alpha1.SchemeBuilder.Build()
+ revision := fromConfiguration(configuration)
for _, configSpec := range component.ConfigTemplates {
var item *appsv1alpha1.ConfigurationItemDetail
cmName := core.GetComponentCfgName(cluster.Name, component.Name, configSpec.Name)
@@ -121,10 +124,42 @@ func (wrapper *renderWrapper) renderConfigTemplate(cluster *appsv1alpha1.Cluster
if err := wrapper.addRenderedObject(configSpec.ComponentTemplateSpec, newCMObj, scheme, configuration); err != nil {
return err
}
+ if err := updateConfigMetaForCM(newCMObj, item, revision); err != nil {
+ return err
+ }
}
return nil
}
+func fromConfiguration(configuration *appsv1alpha1.Configuration) string {
+ if configuration == nil {
+ return ""
+ }
+ return strconv.FormatInt(configuration.GetGeneration(), 10)
+}
+
+func updateConfigMetaForCM(newCMObj *corev1.ConfigMap, item *appsv1alpha1.ConfigurationItemDetail, revision string) (err error) {
+ if item == nil {
+ return
+ }
+
+ annotations := newCMObj.GetAnnotations()
+ if annotations == nil {
+ annotations = make(map[string]string)
+ }
+ b, err := json.Marshal(item)
+ if err != nil {
+ return err
+ }
+ annotations[constant.ConfigAppliedVersionAnnotationKey] = string(b)
+ hash, _ := cfgutil.ComputeHash(newCMObj.Data)
+ annotations[constant.CMInsCurrentConfigurationHashLabelKey] = hash
+ annotations[constant.ConfigurationRevision] = revision
+ annotations[constant.CMConfigurationTemplateVersion] = item.Version
+ newCMObj.Annotations = annotations
+ return
+}
+
func applyUpdatedParameters(item *appsv1alpha1.ConfigurationItemDetail, cm *corev1.ConfigMap, configSpec appsv1alpha1.ComponentConfigSpec, cli client.Client, ctx context.Context) (err error) {
var newData map[string]string
var configConstraint *appsv1alpha1.ConfigConstraint
@@ -309,7 +344,7 @@ func generateConfigMapFromTpl(cluster *appsv1alpha1.Cluster,
}
// Using ConfigMap cue template render to configmap of config
- return factory.BuildConfigMapWithTemplate(cluster, component, configs, cmName, configConstraintName, templateSpec)
+ return factory.BuildConfigMapWithTemplate(cluster, component, configs, cmName, templateSpec), nil
}
// renderConfigMapTemplate renders config file using template engine
diff --git a/internal/controller/configuration/tool_image_builder.go b/internal/controller/configuration/tool_image_builder.go
index 2ab3eed0aef..6a3f046817e 100644
--- a/internal/controller/configuration/tool_image_builder.go
+++ b/internal/controller/configuration/tool_image_builder.go
@@ -20,13 +20,13 @@ along with this program. If not, see .
package configuration
import (
- "github.com/apecloud/kubeblocks/internal/controller/factory"
corev1 "k8s.io/api/core/v1"
appsv1alpha1 "github.com/apecloud/kubeblocks/apis/apps/v1alpha1"
cfgcm "github.com/apecloud/kubeblocks/internal/configuration/config_manager"
"github.com/apecloud/kubeblocks/internal/constant"
"github.com/apecloud/kubeblocks/internal/controller/component"
+ "github.com/apecloud/kubeblocks/internal/controller/factory"
intctrlutil "github.com/apecloud/kubeblocks/internal/controllerutil"
viper "github.com/apecloud/kubeblocks/internal/viperx"
)
diff --git a/internal/controller/factory/builder.go b/internal/controller/factory/builder.go
index 4a03a8aac58..72469c2a08f 100644
--- a/internal/controller/factory/builder.go
+++ b/internal/controller/factory/builder.go
@@ -20,7 +20,6 @@ along with this program. If not, see .
package factory
import (
- "embed"
"encoding/base64"
"encoding/hex"
"encoding/json"
@@ -32,7 +31,6 @@ import (
"github.com/google/uuid"
snapshotv1 "github.com/kubernetes-csi/external-snapshotter/client/v6/apis/volumesnapshot/v1"
- "github.com/leaanthony/debme"
appsv1 "k8s.io/api/apps/v1"
batchv1 "k8s.io/api/batch/v1"
corev1 "k8s.io/api/core/v1"
@@ -42,7 +40,7 @@ import (
"k8s.io/apimachinery/pkg/util/rand"
appsv1alpha1 "github.com/apecloud/kubeblocks/apis/apps/v1alpha1"
- dataprotectionv1alpha1 "github.com/apecloud/kubeblocks/apis/dataprotection/v1alpha1"
+ dpv1alpha1 "github.com/apecloud/kubeblocks/apis/dataprotection/v1alpha1"
workloads "github.com/apecloud/kubeblocks/apis/workloads/v1alpha1"
cfgcm "github.com/apecloud/kubeblocks/internal/configuration/config_manager"
"github.com/apecloud/kubeblocks/internal/constant"
@@ -50,6 +48,7 @@ import (
"github.com/apecloud/kubeblocks/internal/controller/component"
"github.com/apecloud/kubeblocks/internal/controller/rsm"
intctrlutil "github.com/apecloud/kubeblocks/internal/controllerutil"
+ dptypes "github.com/apecloud/kubeblocks/internal/dataprotection/types"
)
const (
@@ -60,53 +59,6 @@ const (
MountPath = "/etc/pki/tls"
)
-var (
- //go:embed cue/*
- cueTemplates embed.FS
- cacheCtx = map[string]interface{}{}
-)
-
-func getCacheCUETplValue(key string, valueCreator func() (*intctrlutil.CUETpl, error)) (*intctrlutil.CUETpl, error) {
- vIf, ok := cacheCtx[key]
- if ok {
- return vIf.(*intctrlutil.CUETpl), nil
- }
- v, err := valueCreator()
- if err != nil {
- return nil, err
- }
- cacheCtx[key] = v
- return v, err
-}
-
-func buildFromCUE(tplName string, fillMap map[string]any, lookupKey string, target any) error {
- cueFS, _ := debme.FS(cueTemplates, "cue")
- cueTpl, err := getCacheCUETplValue(tplName, func() (*intctrlutil.CUETpl, error) {
- return intctrlutil.NewCUETplFromBytes(cueFS.ReadFile(tplName))
- })
- if err != nil {
- return err
- }
- cueValue := intctrlutil.NewCUEBuilder(*cueTpl)
-
- for k, v := range fillMap {
- if err := cueValue.FillObj(k, v); err != nil {
- return err
- }
- }
-
- b, err := cueValue.Lookup(lookupKey)
- if err != nil {
- return err
- }
-
- if err = json.Unmarshal(b, target); err != nil {
- return err
- }
-
- return nil
-}
-
func processContainersInjection(reqCtx intctrlutil.RequestCtx,
cluster *appsv1alpha1.Cluster,
component *component.SynthesizedComponent,
@@ -245,50 +197,14 @@ func BuildPersistentVolumeClaimLabels(component *component.SynthesizedComponent,
}
}
-func BuildSvcListWithCustomAttributes(cluster *appsv1alpha1.Cluster, component *component.SynthesizedComponent,
- customAttributeSetter func(*corev1.Service)) ([]*corev1.Service, error) {
- services, err := BuildSvcList(cluster, component)
- if err != nil {
- return nil, err
- }
- if customAttributeSetter != nil {
- for _, svc := range services {
- customAttributeSetter(svc)
- }
- }
- return services, nil
-}
-
-func BuildSvcList(cluster *appsv1alpha1.Cluster, component *component.SynthesizedComponent) ([]*corev1.Service, error) {
- const tplFile = "service_template.cue"
- var result = make([]*corev1.Service, 0)
- for _, item := range component.Services {
- if len(item.Spec.Ports) == 0 {
- continue
- }
- svc := corev1.Service{}
- if err := buildFromCUE(tplFile, map[string]any{
- "cluster": cluster,
- "service": item,
- "component": component,
- }, "svc", &svc); err != nil {
- return nil, err
- }
- result = append(result, &svc)
- }
- return result, nil
-}
-
-func BuildHeadlessSvc(cluster *appsv1alpha1.Cluster, component *component.SynthesizedComponent) (*corev1.Service, error) {
- const tplFile = "headless_service_template.cue"
- service := corev1.Service{}
- if err := buildFromCUE(tplFile, map[string]any{
- "cluster": cluster,
- "component": component,
- }, "service", &service); err != nil {
- return nil, err
+func BuildCommonLabels(cluster *appsv1alpha1.Cluster,
+ component *component.SynthesizedComponent) map[string]string {
+ return map[string]string{
+ constant.AppManagedByLabelKey: constant.AppName,
+ constant.AppNameLabelKey: component.ClusterDefName,
+ constant.AppInstanceLabelKey: cluster.Name,
+ constant.KBAppComponentLabelKey: component.Name,
}
- return &service, nil
}
func BuildSts(reqCtx intctrlutil.RequestCtx, cluster *appsv1alpha1.Cluster,
@@ -300,12 +216,7 @@ func BuildSts(reqCtx intctrlutil.RequestCtx, cluster *appsv1alpha1.Cluster,
}
}
- commonLabels := map[string]string{
- constant.AppManagedByLabelKey: constant.AppName,
- constant.AppNameLabelKey: component.ClusterDefName,
- constant.AppInstanceLabelKey: cluster.Name,
- constant.KBAppComponentLabelKey: component.Name,
- }
+ commonLabels := BuildCommonLabels(cluster, component)
podBuilder := builder.NewPodBuilder("", "").
AddLabelsInMap(commonLabels).
AddLabels(constant.AppComponentLabelKey, component.CompDefName).
@@ -352,6 +263,15 @@ func BuildSts(reqCtx intctrlutil.RequestCtx, cluster *appsv1alpha1.Cluster,
return sts, nil
}
+func buildWellKnownLabels(clusterDefName, clusterName, componentName string) map[string]string {
+ return map[string]string{
+ constant.AppManagedByLabelKey: constant.AppName,
+ constant.AppNameLabelKey: clusterDefName,
+ constant.AppInstanceLabelKey: clusterName,
+ constant.KBAppComponentLabelKey: componentName,
+ }
+}
+
func BuildRSM(reqCtx intctrlutil.RequestCtx, cluster *appsv1alpha1.Cluster,
component *component.SynthesizedComponent, envConfigName string) (*workloads.ReplicatedStateMachine, error) {
vctToPVC := func(vct corev1.PersistentVolumeClaimTemplate) corev1.PersistentVolumeClaim {
@@ -361,12 +281,7 @@ func BuildRSM(reqCtx intctrlutil.RequestCtx, cluster *appsv1alpha1.Cluster,
}
}
- commonLabels := map[string]string{
- constant.AppManagedByLabelKey: constant.AppName,
- constant.AppNameLabelKey: component.ClusterDefName,
- constant.AppInstanceLabelKey: cluster.Name,
- constant.KBAppComponentLabelKey: component.Name,
- }
+ commonLabels := buildWellKnownLabels(component.ClusterDefName, cluster.Name, component.Name)
addCommonLabels := func(service *corev1.Service) {
if service == nil {
return
@@ -546,6 +461,10 @@ func separateServices(services []corev1.Service) (*corev1.Service, []corev1.Serv
}
func buildRoleInfo(component *component.SynthesizedComponent) ([]workloads.ReplicaRole, *workloads.RoleProbe, *workloads.MembershipReconfiguration, *workloads.MemberUpdateStrategy) {
+ if component.RSMSpec != nil {
+ return buildRoleInfo2(component)
+ }
+
var (
roles []workloads.ReplicaRole
probe *workloads.RoleProbe
@@ -562,6 +481,7 @@ func buildRoleInfo(component *component.SynthesizedComponent) ([]workloads.Repli
probe.FailureThreshold = roleProbe.FailureThreshold
// set to default value
probe.SuccessThreshold = 1
+ probe.RoleUpdateMechanism = workloads.DirectAPIServerEventUpdate
}
// TODO(free6om): set default reconfiguration actions after relative addon refactored
@@ -580,6 +500,11 @@ func buildRoleInfo(component *component.SynthesizedComponent) ([]workloads.Repli
return roles, probe, reconfiguration, strategy
}
+func buildRoleInfo2(component *component.SynthesizedComponent) ([]workloads.ReplicaRole, *workloads.RoleProbe, *workloads.MembershipReconfiguration, *workloads.MemberUpdateStrategy) {
+ rsmSpec := component.RSMSpec
+ return rsmSpec.Roles, rsmSpec.RoleProbe, rsmSpec.MembershipReconfiguration, rsmSpec.MemberUpdateStrategy
+}
+
func buildRoleInfoFromReplication() []workloads.ReplicaRole {
return []workloads.ReplicaRole{
{
@@ -706,7 +631,7 @@ func buildActionFromCharacterType(characterType string, isConsensus bool) []work
{
Image: "registry.cn-hangzhou.aliyuncs.com/apecloud/mongo:5.0.14",
Command: []string{
- "Status=$(export CLIENT=`which mongosh>/dev/null&&echo mongosh||echo mongo`; $CLIENT -u $KB_RSM_USERNAME -p $KB_RSM_PASSWORD 127.0.0.1:27017 --quiet --eval \"JSON.stringify(rs.status())\") &&",
+ "Status=$(export CLIENT=`which mongosh>/dev/null&&echo mongosh||echo mongo`; $CLIENT -u $KB_RSM_USERNAME -p $KB_RSM_PASSWORD 127.0.0.1:27017 --authenticationDatabase admin --quiet --eval \"JSON.stringify(rs.status())\") &&",
"MyState=$(echo $Status | jq '.myState') &&",
"echo $Status | jq \".members[] | select(.state == ($MyState | tonumber)) | .stateStr\" |tr '[:upper:]' '[:lower:]' | xargs echo -n",
},
@@ -743,19 +668,19 @@ func randomString(length int) string {
}
func BuildConnCredential(clusterDefinition *appsv1alpha1.ClusterDefinition, cluster *appsv1alpha1.Cluster,
- component *component.SynthesizedComponent) (*corev1.Secret, error) {
- const tplFile = "conn_credential_template.cue"
-
- connCredential := corev1.Secret{}
- if err := buildFromCUE(tplFile, map[string]any{
- "clusterdefinition": clusterDefinition,
- "cluster": cluster,
- }, "secret", &connCredential); err != nil {
- return nil, err
+ component *component.SynthesizedComponent) *corev1.Secret {
+ wellKnownLabels := buildWellKnownLabels(clusterDefinition.Name, cluster.Name, "")
+ delete(wellKnownLabels, constant.KBAppComponentLabelKey)
+ credentialBuilder := builder.NewSecretBuilder(cluster.Namespace, fmt.Sprintf("%s-conn-credential", cluster.Name)).
+ AddLabelsInMap(wellKnownLabels).
+ SetStringData(clusterDefinition.Spec.ConnectionCredential)
+ if len(clusterDefinition.Spec.Type) > 0 {
+ credentialBuilder.AddLabels("apps.kubeblocks.io/cluster-type", clusterDefinition.Spec.Type)
}
+ connCredential := credentialBuilder.GetObject()
if len(connCredential.StringData) == 0 {
- return &connCredential, nil
+ return connCredential
}
replaceVarObjects := func(k, v *string, i int, origValue string, varObjectsMap map[string]string) {
@@ -821,65 +746,49 @@ func BuildConnCredential(clusterDefinition *appsv1alpha1.ClusterDefinition, clus
m[fmt.Sprintf("$(CONN_CREDENTIAL).%s", k)] = v
}
replaceData(m)
- return &connCredential, nil
-}
-
-func BuildPDB(cluster *appsv1alpha1.Cluster, component *component.SynthesizedComponent) (*policyv1.PodDisruptionBudget, error) {
- const tplFile = "pdb_template.cue"
- pdb := policyv1.PodDisruptionBudget{}
- if err := buildFromCUE(tplFile, map[string]any{
- "cluster": cluster,
- "component": component,
- }, "pdb", &pdb); err != nil {
- return nil, err
- }
- return &pdb, nil
+ return connCredential
}
-func BuildDeploy(reqCtx intctrlutil.RequestCtx, cluster *appsv1alpha1.Cluster, component *component.SynthesizedComponent, envConfigName string) (*appsv1.Deployment, error) {
- const tplFile = "deployment_template.cue"
- deploy := appsv1.Deployment{}
- if err := buildFromCUE(tplFile, map[string]any{
- "cluster": cluster,
- "component": component,
- }, "deployment", &deploy); err != nil {
- return nil, err
- }
-
- if component.StatelessSpec != nil {
- deploy.Spec.Strategy = component.StatelessSpec.UpdateStrategy
- }
- if err := processContainersInjection(reqCtx, cluster, component, envConfigName, &deploy.Spec.Template.Spec); err != nil {
- return nil, err
- }
- return &deploy, nil
+func BuildPDB(cluster *appsv1alpha1.Cluster, component *component.SynthesizedComponent) *policyv1.PodDisruptionBudget {
+ wellKnownLabels := buildWellKnownLabels(component.ClusterDefName, cluster.Name, component.Name)
+ return builder.NewPDBBuilder(cluster.Namespace, fmt.Sprintf("%s-%s", cluster.Name, component.Name)).
+ AddLabelsInMap(wellKnownLabels).
+ AddLabels(constant.AppComponentLabelKey, component.CompDefName).
+ AddSelectorsInMap(wellKnownLabels).
+ GetObject()
}
func BuildPVC(cluster *appsv1alpha1.Cluster,
component *component.SynthesizedComponent,
vct *corev1.PersistentVolumeClaimTemplate,
pvcKey types.NamespacedName,
- snapshotName string) (*corev1.PersistentVolumeClaim, error) {
- pvc := corev1.PersistentVolumeClaim{}
- if err := buildFromCUE("pvc_template.cue", map[string]any{
- "cluster": cluster,
- "component": component,
- "volumeClaimTemplate": vct,
- "pvc_key": pvcKey,
- "snapshot_name": snapshotName,
- }, "pvc", &pvc); err != nil {
- return nil, err
+ snapshotName string) *corev1.PersistentVolumeClaim {
+ wellKnownLabels := buildWellKnownLabels(component.ClusterDefName, cluster.Name, component.Name)
+ pvcBuilder := builder.NewPVCBuilder(pvcKey.Namespace, pvcKey.Name).
+ AddLabelsInMap(wellKnownLabels).
+ AddLabels(constant.VolumeClaimTemplateNameLabelKey, vct.Name).
+ SetAccessModes(vct.Spec.AccessModes).
+ SetResources(vct.Spec.Resources)
+ if vct.Spec.StorageClassName != nil {
+ pvcBuilder.SetStorageClass(*vct.Spec.StorageClassName)
+ }
+ if len(snapshotName) > 0 {
+ apiGroup := "snapshot.storage.k8s.io"
+ pvcBuilder.SetDataSource(corev1.TypedLocalObjectReference{
+ APIGroup: &apiGroup,
+ Kind: "VolumeSnapshot",
+ Name: snapshotName,
+ })
}
- BuildPersistentVolumeClaimLabels(component, &pvc, vct.Name)
- return &pvc, nil
+ pvc := pvcBuilder.GetObject()
+ BuildPersistentVolumeClaimLabels(component, pvc, vct.Name)
+ return pvc
}
// BuildEnvConfig builds cluster component context ConfigMap object, which is to be used in workload container's
// envFrom.configMapRef with name of "$(cluster.metadata.name)-$(component.name)-env" pattern.
-func BuildEnvConfig(cluster *appsv1alpha1.Cluster, component *component.SynthesizedComponent) (*corev1.ConfigMap, error) {
- const tplFile = "env_config_template.cue"
+func BuildEnvConfig(cluster *appsv1alpha1.Cluster, component *component.SynthesizedComponent) *corev1.ConfigMap {
envData := map[string]string{}
-
// add component envs
if component.ComponentRefEnvs != nil {
for _, env := range component.ComponentRefEnvs {
@@ -887,167 +796,145 @@ func BuildEnvConfig(cluster *appsv1alpha1.Cluster, component *component.Synthesi
}
}
- config := corev1.ConfigMap{}
- if err := buildFromCUE(tplFile, map[string]any{
- "cluster": cluster,
- "component": component,
- "config.data": envData,
- }, "config", &config); err != nil {
- return nil, err
- }
- return &config, nil
+ wellKnownLabels := buildWellKnownLabels(component.ClusterDefName, cluster.Name, component.Name)
+ wellKnownLabels[constant.AppComponentLabelKey] = component.CompDefName
+ return builder.NewConfigMapBuilder(cluster.Namespace, fmt.Sprintf("%s-%s-env", cluster.Name, component.Name)).
+ AddLabelsInMap(wellKnownLabels).
+ AddLabels(constant.AppConfigTypeLabelKey, "kubeblocks-env").
+ SetData(envData).
+ GetObject()
}
func BuildBackup(cluster *appsv1alpha1.Cluster,
component *component.SynthesizedComponent,
backupPolicyName string,
backupKey types.NamespacedName,
- backupType string) (*dataprotectionv1alpha1.Backup, error) {
- backup := dataprotectionv1alpha1.Backup{}
- if err := buildFromCUE("backup_job_template.cue", map[string]any{
- "cluster": cluster,
- "component": component,
- "backupPolicyName": backupPolicyName,
- "backupJobKey": backupKey,
- "backupType": backupType,
- }, "backupJob", &backup); err != nil {
- return nil, err
- }
- return &backup, nil
+ backupMethod string) *dpv1alpha1.Backup {
+ return builder.NewBackupBuilder(backupKey.Namespace, backupKey.Name).
+ AddLabels(dptypes.DataProtectionLabelBackupMethodKey, backupMethod).
+ AddLabels(dptypes.DataProtectionLabelBackupPolicyKey, backupPolicyName).
+ AddLabels(constant.KBManagedByKey, "cluster").
+ AddLabels(constant.AppNameLabelKey, component.ClusterDefName).
+ AddLabels(constant.AppInstanceLabelKey, cluster.Name).
+ AddLabels(constant.AppManagedByLabelKey, constant.AppName).
+ AddLabels(constant.KBAppComponentLabelKey, component.Name).
+ SetBackupPolicyName(backupPolicyName).
+ SetBackupMethod(backupMethod).
+ GetObject()
}
func BuildConfigMapWithTemplate(cluster *appsv1alpha1.Cluster,
component *component.SynthesizedComponent,
configs map[string]string,
cmName string,
- configConstraintName string,
- configTemplateSpec appsv1alpha1.ComponentTemplateSpec) (*corev1.ConfigMap, error) {
- const tplFile = "config_template.cue"
- cueFS, _ := debme.FS(cueTemplates, "cue")
- cueTpl, err := getCacheCUETplValue(tplFile, func() (*intctrlutil.CUETpl, error) {
- return intctrlutil.NewCUETplFromBytes(cueFS.ReadFile(tplFile))
- })
- if err != nil {
- return nil, err
- }
-
- cueValue := intctrlutil.NewCUEBuilder(*cueTpl)
- // prepare cue data
- configMeta := map[string]map[string]string{
- "clusterDefinition": {
- "name": cluster.Spec.ClusterDefRef,
- },
- "cluster": {
- "name": cluster.GetName(),
- "namespace": cluster.GetNamespace(),
- },
- "component": {
- "name": component.Name,
- "compDefName": component.CompDefName,
- "characterType": component.CharacterType,
- "configName": cmName,
- "templateName": configTemplateSpec.TemplateRef,
- "configConstraintsName": configConstraintName,
- "configTemplateName": configTemplateSpec.Name,
- },
- }
- configBytes, err := json.Marshal(configMeta)
- if err != nil {
- return nil, err
- }
-
- // Generate config files context by rendering cue template
- if err = cueValue.Fill("meta", configBytes); err != nil {
- return nil, err
- }
-
- configStrByte, err := cueValue.Lookup("config")
- if err != nil {
- return nil, err
- }
-
- cm := corev1.ConfigMap{}
- if err = json.Unmarshal(configStrByte, &cm); err != nil {
- return nil, err
- }
-
- // Update rendered config
- cm.Data = configs
- return &cm, nil
+ configTemplateSpec appsv1alpha1.ComponentTemplateSpec) *corev1.ConfigMap {
+ wellKnownLabels := buildWellKnownLabels(component.ClusterDefName, cluster.Name, component.Name)
+ wellKnownLabels[constant.AppComponentLabelKey] = component.CompDefName
+ return builder.NewConfigMapBuilder(cluster.Namespace, cmName).
+ AddLabelsInMap(wellKnownLabels).
+ AddLabels(constant.CMConfigurationTypeLabelKey, constant.ConfigInstanceType).
+ AddLabels(constant.CMTemplateNameLabelKey, configTemplateSpec.TemplateRef).
+ AddAnnotations(constant.DisableUpgradeInsConfigurationAnnotationKey, strconv.FormatBool(false)).
+ SetData(configs).
+ GetObject()
}
func BuildCfgManagerContainer(sidecarRenderedParam *cfgcm.CfgManagerBuildParams, component *component.SynthesizedComponent) (*corev1.Container, error) {
- const tplFile = "config_manager_sidecar.cue"
- cueFS, _ := debme.FS(cueTemplates, "cue")
- cueTpl, err := getCacheCUETplValue(tplFile, func() (*intctrlutil.CUETpl, error) {
- return intctrlutil.NewCUETplFromBytes(cueFS.ReadFile(tplFile))
+ var env []corev1.EnvVar
+ env = append(env, corev1.EnvVar{
+ Name: "CONFIG_MANAGER_POD_IP",
+ ValueFrom: &corev1.EnvVarSource{
+ FieldRef: &corev1.ObjectFieldSelector{
+ APIVersion: "v1",
+ FieldPath: "status.podIP",
+ },
+ },
})
- if err != nil {
- return nil, err
- }
-
- cueValue := intctrlutil.NewCUEBuilder(*cueTpl)
- paramBytes, err := json.Marshal(sidecarRenderedParam)
- if err != nil {
- return nil, err
- }
-
- if err = cueValue.Fill("parameter", paramBytes); err != nil {
- return nil, err
- }
-
- containerStrByte, err := cueValue.Lookup("template")
- if err != nil {
- return nil, err
+ if len(sidecarRenderedParam.CharacterType) > 0 {
+ env = append(env, corev1.EnvVar{
+ Name: "DB_TYPE",
+ Value: sidecarRenderedParam.CharacterType,
+ })
}
- container := corev1.Container{}
- if err = json.Unmarshal(containerStrByte, &container); err != nil {
- return nil, err
+ if sidecarRenderedParam.CharacterType == "mysql" {
+ env = append(env, corev1.EnvVar{
+ Name: "MYSQL_USER",
+ ValueFrom: &corev1.EnvVarSource{
+ SecretKeyRef: &corev1.SecretKeySelector{
+ Key: "username",
+ LocalObjectReference: corev1.LocalObjectReference{Name: sidecarRenderedParam.SecreteName},
+ },
+ },
+ },
+ corev1.EnvVar{
+ Name: "MYSQL_PASSWORD",
+ ValueFrom: &corev1.EnvVarSource{
+ SecretKeyRef: &corev1.SecretKeySelector{
+ Key: "password",
+ LocalObjectReference: corev1.LocalObjectReference{Name: sidecarRenderedParam.SecreteName},
+ },
+ },
+ },
+ corev1.EnvVar{
+ Name: "DATA_SOURCE_NAME",
+ Value: "$(MYSQL_USER):$(MYSQL_PASSWORD)@(localhost:3306)/",
+ },
+ )
+ }
+ containerBuilder := builder.NewContainerBuilder(sidecarRenderedParam.ManagerName).
+ AddCommands("env").
+ AddArgs("PATH=/usr/local/sbin:/usr/local/bin:/usr/sbin:/usr/bin:/sbin:/bin:$(TOOLS_PATH)").
+ AddArgs("/bin/reloader").
+ AddArgs(sidecarRenderedParam.Args...).
+ AddEnv(env...).
+ SetImage(sidecarRenderedParam.Image).
+ SetImagePullPolicy(corev1.PullIfNotPresent).
+ AddVolumeMounts(sidecarRenderedParam.Volumes...)
+ if sidecarRenderedParam.ShareProcessNamespace {
+ user := int64(0)
+ containerBuilder.SetSecurityContext(corev1.SecurityContext{
+ RunAsUser: &user,
+ })
}
+ container := containerBuilder.GetObject()
- if err := injectEnvs(sidecarRenderedParam.Cluster, component, sidecarRenderedParam.EnvConfigName, &container); err != nil {
+ if err := injectEnvs(sidecarRenderedParam.Cluster, component, sidecarRenderedParam.EnvConfigName, container); err != nil {
return nil, err
}
- intctrlutil.InjectZeroResourcesLimitsIfEmpty(&container)
- return &container, nil
-}
-
-func BuildBackupManifestsJob(key types.NamespacedName, backup *dataprotectionv1alpha1.Backup, podSpec *corev1.PodSpec) (*batchv1.Job, error) {
- const tplFile = "backup_manifests_template.cue"
- job := &batchv1.Job{}
- if err := buildFromCUE(tplFile,
- map[string]any{
- "job.metadata.name": key.Name,
- "job.metadata.namespace": key.Namespace,
- "backup": backup,
- "podSpec": podSpec,
- },
- "job", job); err != nil {
- return nil, err
- }
- return job, nil
+ intctrlutil.InjectZeroResourcesLimitsIfEmpty(container)
+ return container, nil
}
func BuildRestoreJob(cluster *appsv1alpha1.Cluster, synthesizedComponent *component.SynthesizedComponent, name, image string, command []string,
volumes []corev1.Volume, volumeMounts []corev1.VolumeMount, env []corev1.EnvVar, resources *corev1.ResourceRequirements) (*batchv1.Job, error) {
- const tplFile = "restore_job_template.cue"
- job := &batchv1.Job{}
- fillMaps := map[string]any{
- "job.metadata.name": name,
- "job.metadata.namespace": cluster.Namespace,
- "job.spec.template.spec.volumes": volumes,
- "container.image": image,
- "container.command": command,
- "container.volumeMounts": volumeMounts,
- "container.env": env,
- }
+ containerBuilder := builder.NewContainerBuilder("restore").
+ SetImage(image).
+ SetImagePullPolicy(corev1.PullIfNotPresent).
+ AddCommands(command...).
+ AddVolumeMounts(volumeMounts...).
+ AddEnv(env...)
if resources != nil {
- fillMaps["container.resources"] = *resources
+ containerBuilder.SetResources(*resources)
+ }
+ container := containerBuilder.GetObject()
+
+ ctx := corev1.PodSecurityContext{}
+ user := int64(0)
+ ctx.RunAsUser = &user
+ pod := builder.NewPodBuilder(cluster.Namespace, "").
+ AddContainer(*container).
+ AddVolumes(volumes...).
+ SetRestartPolicy(corev1.RestartPolicyOnFailure).
+ SetSecurityContext(ctx).
+ GetObject()
+ template := corev1.PodTemplateSpec{
+ Spec: pod.Spec,
}
- if err := buildFromCUE(tplFile, fillMaps, "job", job); err != nil {
- return nil, err
- }
+ job := builder.NewJobBuilder(cluster.Namespace, name).
+ AddLabels(constant.AppManagedByLabelKey, constant.AppName).
+ SetPodTemplateSpec(template).
+ GetObject()
containers := job.Spec.Template.Spec.Containers
if len(containers) > 0 {
if err := injectEnvs(cluster, synthesizedComponent, "", &containers[0]); err != nil {
@@ -1066,16 +953,14 @@ func BuildRestoreJob(cluster *appsv1alpha1.Cluster, synthesizedComponent *compon
func BuildCfgManagerToolsContainer(sidecarRenderedParam *cfgcm.CfgManagerBuildParams, component *component.SynthesizedComponent, toolsMetas []appsv1alpha1.ToolConfig, toolsMap map[string]cfgcm.ConfigSpecMeta) ([]corev1.Container, error) {
toolContainers := make([]corev1.Container, 0, len(toolsMetas))
for _, toolConfig := range toolsMetas {
- toolContainer := corev1.Container{
- Name: toolConfig.Name,
- Command: toolConfig.Command,
- ImagePullPolicy: corev1.PullIfNotPresent,
- VolumeMounts: sidecarRenderedParam.Volumes,
- }
- if toolConfig.Image != "" {
- toolContainer.Image = toolConfig.Image
+ toolContainerBuilder := builder.NewContainerBuilder(toolConfig.Name).
+ AddCommands(toolConfig.Command...).
+ SetImagePullPolicy(corev1.PullIfNotPresent).
+ AddVolumeMounts(sidecarRenderedParam.Volumes...)
+ if len(toolConfig.Image) > 0 {
+ toolContainerBuilder.SetImage(toolConfig.Image)
}
- toolContainers = append(toolContainers, toolContainer)
+ toolContainers = append(toolContainers, *toolContainerBuilder.GetObject())
}
for i := range toolContainers {
container := &toolContainers[i]
@@ -1097,39 +982,54 @@ func setToolsScriptsPath(container *corev1.Container, meta cfgcm.ConfigSpecMeta)
})
}
-func BuildVolumeSnapshotClass(name string, driver string) (*snapshotv1.VolumeSnapshotClass, error) {
- const tplFile = "volumesnapshotclass.cue"
- vsc := &snapshotv1.VolumeSnapshotClass{}
- if err := buildFromCUE(tplFile,
- map[string]any{
- "class.metadata.name": name,
- "class.driver": driver,
- },
- "class", vsc); err != nil {
- return nil, err
- }
- return vsc, nil
-}
-
-func BuildServiceAccount(cluster *appsv1alpha1.Cluster) (*corev1.ServiceAccount, error) {
- return buildRBACObject[corev1.ServiceAccount](cluster, "serviceaccount")
+func BuildVolumeSnapshotClass(name string, driver string) *snapshotv1.VolumeSnapshotClass {
+ return builder.NewVolumeSnapshotClassBuilder("", name).
+ AddLabels(constant.AppManagedByLabelKey, constant.AppName).
+ SetDriver(driver).
+ SetDeletionPolicy(snapshotv1.VolumeSnapshotContentDelete).
+ GetObject()
}
-func BuildRoleBinding(cluster *appsv1alpha1.Cluster) (*rbacv1.RoleBinding, error) {
- return buildRBACObject[rbacv1.RoleBinding](cluster, "rolebinding")
+func BuildServiceAccount(cluster *appsv1alpha1.Cluster) *corev1.ServiceAccount {
+ wellKnownLabels := buildWellKnownLabels(cluster.Spec.ClusterDefRef, cluster.Name, "")
+ delete(wellKnownLabels, constant.KBAppComponentLabelKey)
+ return builder.NewServiceAccountBuilder(cluster.Namespace, fmt.Sprintf("kb-%s", cluster.Name)).
+ AddLabelsInMap(wellKnownLabels).
+ GetObject()
}
-func BuildClusterRoleBinding(cluster *appsv1alpha1.Cluster) (*rbacv1.ClusterRoleBinding, error) {
- return buildRBACObject[rbacv1.ClusterRoleBinding](cluster, "clusterrolebinding")
+func BuildRoleBinding(cluster *appsv1alpha1.Cluster) *rbacv1.RoleBinding {
+ wellKnownLabels := buildWellKnownLabels(cluster.Spec.ClusterDefRef, cluster.Name, "")
+ delete(wellKnownLabels, constant.KBAppComponentLabelKey)
+ return builder.NewRoleBindingBuilder(cluster.Namespace, fmt.Sprintf("kb-%s", cluster.Name)).
+ AddLabelsInMap(wellKnownLabels).
+ SetRoleRef(rbacv1.RoleRef{
+ APIGroup: rbacv1.GroupName,
+ Kind: "ClusterRole",
+ Name: constant.RBACRoleName,
+ }).
+ AddSubjects(rbacv1.Subject{
+ Kind: rbacv1.ServiceAccountKind,
+ Namespace: cluster.Namespace,
+ Name: fmt.Sprintf("kb-%s", cluster.Name),
+ }).
+ GetObject()
}
-func buildRBACObject[Tp corev1.ServiceAccount | rbacv1.RoleBinding | rbacv1.ClusterRoleBinding](
- cluster *appsv1alpha1.Cluster, key string) (*Tp, error) {
- const tplFile = "rbac_template.cue"
- var obj Tp
- pObj := &obj
- if err := buildFromCUE(tplFile, map[string]any{"cluster": cluster}, key, pObj); err != nil {
- return nil, err
- }
- return pObj, nil
+func BuildClusterRoleBinding(cluster *appsv1alpha1.Cluster) *rbacv1.ClusterRoleBinding {
+ wellKnownLabels := buildWellKnownLabels(cluster.Spec.ClusterDefRef, cluster.Name, "")
+ delete(wellKnownLabels, constant.KBAppComponentLabelKey)
+ return builder.NewClusterRoleBindingBuilder(cluster.Namespace, fmt.Sprintf("kb-%s", cluster.Name)).
+ AddLabelsInMap(wellKnownLabels).
+ SetRoleRef(rbacv1.RoleRef{
+ APIGroup: rbacv1.GroupName,
+ Kind: "ClusterRole",
+ Name: constant.RBACClusterRoleName,
+ }).
+ AddSubjects(rbacv1.Subject{
+ Kind: rbacv1.ServiceAccountKind,
+ Namespace: cluster.Namespace,
+ Name: fmt.Sprintf("kb-%s", cluster.Name),
+ }).
+ GetObject()
}
diff --git a/internal/controller/factory/builder_test.go b/internal/controller/factory/builder_test.go
index 8f3807ebd0e..341beba304f 100644
--- a/internal/controller/factory/builder_test.go
+++ b/internal/controller/factory/builder_test.go
@@ -22,22 +22,18 @@ package factory
import (
"encoding/json"
"fmt"
- "testing"
. "github.com/onsi/ginkgo/v2"
. "github.com/onsi/gomega"
- "github.com/leaanthony/debme"
"golang.org/x/exp/slices"
appsv1 "k8s.io/api/apps/v1"
corev1 "k8s.io/api/core/v1"
metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
"k8s.io/apimachinery/pkg/types"
- ctrl "sigs.k8s.io/controller-runtime"
"sigs.k8s.io/controller-runtime/pkg/client"
appsv1alpha1 "github.com/apecloud/kubeblocks/apis/apps/v1alpha1"
- dataprotectionv1alpha1 "github.com/apecloud/kubeblocks/apis/dataprotection/v1alpha1"
workloads "github.com/apecloud/kubeblocks/apis/workloads/v1alpha1"
cfgcm "github.com/apecloud/kubeblocks/internal/configuration/config_manager"
"github.com/apecloud/kubeblocks/internal/constant"
@@ -47,20 +43,6 @@ import (
viper "github.com/apecloud/kubeblocks/internal/viperx"
)
-var tlog = ctrl.Log.WithName("builder_testing")
-
-func TestReadCUETplFromEmbeddedFS(t *testing.T) {
- cueFS, err := debme.FS(cueTemplates, "cue")
- if err != nil {
- t.Error("Expected no error", err)
- }
- cueTpl, err := intctrlutil.NewCUETplFromBytes(cueFS.ReadFile("conn_credential_template.cue"))
- if err != nil {
- t.Error("Expected no error", err)
- }
- tlog.Info("", "cueValue", cueTpl)
-}
-
var _ = Describe("builder", func() {
const clusterDefName = "test-clusterdef"
const clusterVersionName = "test-clusterversion"
@@ -191,8 +173,7 @@ var _ = Describe("builder", func() {
Namespace: "default",
Name: "data-mysql-01-replicasets-0",
}
- pvc, err := BuildPVC(cluster, synthesizedComponent, &synthesizedComponent.VolumeClaimTemplates[0], pvcKey, snapshotName)
- Expect(err).Should(BeNil())
+ pvc := BuildPVC(cluster, synthesizedComponent, &synthesizedComponent.VolumeClaimTemplates[0], pvcKey, snapshotName)
Expect(pvc).ShouldNot(BeNil())
Expect(pvc.Spec.AccessModes).Should(Equal(sts.Spec.VolumeClaimTemplates[0].Spec.AccessModes))
Expect(pvc.Spec.Resources).Should(Equal(synthesizedComponent.VolumeClaimTemplates[0].Spec.Resources))
@@ -200,27 +181,18 @@ var _ = Describe("builder", func() {
Expect(pvc.Labels[constant.VolumeTypeLabelKey]).ShouldNot(BeEmpty())
})
- It("builds Service correctly", func() {
- _, cluster, synthesizedComponent := newClusterObjs(nil)
- svcList, err := BuildSvcListWithCustomAttributes(cluster, synthesizedComponent, nil)
- Expect(err).Should(BeNil())
- Expect(svcList).ShouldNot(BeEmpty())
- })
-
It("builds Conn. Credential correctly", func() {
var (
clusterDefObj = testapps.NewClusterDefFactoryWithConnCredential("conn-cred").GetObject()
clusterDef, cluster, synthesizedComponent = newClusterObjs(clusterDefObj)
)
- credential, err := BuildConnCredential(clusterDef, cluster, synthesizedComponent)
- Expect(err).Should(BeNil())
+ credential := BuildConnCredential(clusterDef, cluster, synthesizedComponent)
Expect(credential).ShouldNot(BeNil())
Expect(credential.Labels["apps.kubeblocks.io/cluster-type"]).Should(BeEmpty())
By("setting type")
characterType := "test-character-type"
clusterDef.Spec.Type = characterType
- credential, err = BuildConnCredential(clusterDef, cluster, synthesizedComponent)
- Expect(err).Should(BeNil())
+ credential = BuildConnCredential(clusterDef, cluster, synthesizedComponent)
Expect(credential).ShouldNot(BeNil())
Expect(credential.Labels["apps.kubeblocks.io/cluster-type"]).Should(Equal(characterType))
// "username": "root",
@@ -402,18 +374,9 @@ var _ = Describe("builder", func() {
Expect(*rsm.Spec.MemberUpdateStrategy).Should(BeEquivalentTo(workloads.BestEffortParallelUpdateStrategy))
})
- It("builds Deploy correctly", func() {
- reqCtx := newReqCtx()
- _, cluster, synthesizedComponent := newClusterObjs(nil)
- deploy, err := BuildDeploy(reqCtx, cluster, synthesizedComponent, "")
- Expect(err).Should(BeNil())
- Expect(deploy).ShouldNot(BeNil())
- })
-
It("builds PDB correctly", func() {
_, cluster, synthesizedComponent := newClusterObjs(nil)
- pdb, err := BuildPDB(cluster, synthesizedComponent)
- Expect(err).Should(BeNil())
+ pdb := BuildPDB(cluster, synthesizedComponent)
Expect(pdb).ShouldNot(BeNil())
})
@@ -424,8 +387,7 @@ var _ = Describe("builder", func() {
Name: "test-backup-job",
}
backupPolicyName := "test-backup-policy"
- backupJob, err := BuildBackup(cluster, synthesizedComponent, backupPolicyName, backupJobKey, "snapshot")
- Expect(err).Should(BeNil())
+ backupJob := BuildBackup(cluster, synthesizedComponent, backupPolicyName, backupJobKey, "snapshot")
Expect(backupJob).ShouldNot(BeNil())
})
@@ -439,15 +401,14 @@ var _ = Describe("builder", func() {
},
ConfigConstraintRef: "test-config-constraint",
}
- configmap, err := BuildConfigMapWithTemplate(cluster, synthesizedComponent, config,
- "test-cm", tplCfg.ConfigConstraintRef, tplCfg.ComponentTemplateSpec)
- Expect(err).Should(BeNil())
+ configmap := BuildConfigMapWithTemplate(cluster, synthesizedComponent, config,
+ "test-cm", tplCfg.ComponentTemplateSpec)
Expect(configmap).ShouldNot(BeNil())
})
It("builds config manager sidecar container correctly", func() {
_, cluster, synthesizedComponent := newClusterObjs(nil)
- cfg, err := BuildEnvConfig(cluster, synthesizedComponent)
+ cfg := BuildEnvConfig(cluster, synthesizedComponent)
sidecarRenderedParam := &cfgcm.CfgManagerBuildParams{
ManagerName: "cfgmgr",
SecreteName: "test-secret",
@@ -460,7 +421,6 @@ var _ = Describe("builder", func() {
Volumes: []corev1.VolumeMount{},
Cluster: cluster,
}
- Expect(err).Should(BeNil())
configmap, err := BuildCfgManagerContainer(sidecarRenderedParam, synthesizedComponent)
Expect(err).Should(BeNil())
Expect(configmap).ShouldNot(BeNil())
@@ -488,22 +448,6 @@ var _ = Describe("builder", func() {
Expect(*configmap.SecurityContext.RunAsUser).Should(BeEquivalentTo(int64(0)))
})
- It("builds backup manifests job correctly", func() {
- backup := &dataprotectionv1alpha1.Backup{}
- podSpec := &corev1.PodSpec{
- Containers: []corev1.Container{
- {
- Command: []string{"sh"},
- },
- },
- }
- key := types.NamespacedName{Name: "backup", Namespace: "default"}
- job, err := BuildBackupManifestsJob(key, backup, podSpec)
- Expect(err).Should(BeNil())
- Expect(job).ShouldNot(BeNil())
- Expect(job.Name).Should(Equal(key.Name))
- })
-
It("builds restore job correctly", func() {
key := types.NamespacedName{Name: "restore", Namespace: "default"}
volumes := []corev1.Volume{}
@@ -534,22 +478,12 @@ var _ = Describe("builder", func() {
It("builds volume snapshot class correctly", func() {
className := "vsc-test"
driverName := "csi-driver-test"
- obj, err := BuildVolumeSnapshotClass(className, driverName)
- Expect(err).Should(BeNil())
+ obj := BuildVolumeSnapshotClass(className, driverName)
Expect(obj).ShouldNot(BeNil())
Expect(obj.Name).Should(Equal(className))
Expect(obj.Driver).Should(Equal(driverName))
})
- It("builds headless svc correctly", func() {
- _, cluster, synthesizedComponent := newClusterObjs(nil)
- expectSvcName := fmt.Sprintf("%s-%s-headless", cluster.Name, synthesizedComponent.Name)
- obj, err := BuildHeadlessSvc(cluster, synthesizedComponent)
- Expect(err).Should(BeNil())
- Expect(obj).ShouldNot(BeNil())
- Expect(obj.Name).Should(Equal(expectSvcName))
- })
-
It("builds cfg manager tools correctly", func() {
_, cluster, synthesizedComponent := newClusterObjs(nil)
cfgManagerParams := &cfgcm.CfgManagerBuildParams{
@@ -572,8 +506,7 @@ var _ = Describe("builder", func() {
It("builds serviceaccount correctly", func() {
_, cluster, _ := newClusterObjs(nil)
expectName := fmt.Sprintf("kb-%s", cluster.Name)
- sa, err := BuildServiceAccount(cluster)
- Expect(err).Should(BeNil())
+ sa := BuildServiceAccount(cluster)
Expect(sa).ShouldNot(BeNil())
Expect(sa.Name).Should(Equal(expectName))
})
@@ -581,8 +514,7 @@ var _ = Describe("builder", func() {
It("builds rolebinding correctly", func() {
_, cluster, _ := newClusterObjs(nil)
expectName := fmt.Sprintf("kb-%s", cluster.Name)
- rb, err := BuildRoleBinding(cluster)
- Expect(err).Should(BeNil())
+ rb := BuildRoleBinding(cluster)
Expect(rb).ShouldNot(BeNil())
Expect(rb.Name).Should(Equal(expectName))
})
@@ -590,8 +522,7 @@ var _ = Describe("builder", func() {
It("builds clusterrolebinding correctly", func() {
_, cluster, _ := newClusterObjs(nil)
expectName := fmt.Sprintf("kb-%s", cluster.Name)
- crb, err := BuildClusterRoleBinding(cluster)
- Expect(err).Should(BeNil())
+ crb := BuildClusterRoleBinding(cluster)
Expect(crb).ShouldNot(BeNil())
Expect(crb.Name).Should(Equal(expectName))
})
diff --git a/internal/controller/factory/cue/backup_job_template.cue b/internal/controller/factory/cue/backup_job_template.cue
deleted file mode 100644
index 6fa6777d33f..00000000000
--- a/internal/controller/factory/cue/backup_job_template.cue
+++ /dev/null
@@ -1,53 +0,0 @@
-// Copyright (C) 2022-2023 ApeCloud Co., Ltd
-//
-// This file is part of KubeBlocks project
-//
-// This program is free software: you can redistribute it and/or modify
-// it under the terms of the GNU Affero General Public License as published by
-// the Free Software Foundation, either version 3 of the License, or
-// (at your option) any later version.
-//
-// This program is distributed in the hope that it will be useful
-// but WITHOUT ANY WARRANTY; without even the implied warranty of
-// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
-// GNU Affero General Public License for more details.
-//
-// You should have received a copy of the GNU Affero General Public License
-// along with this program. If not, see .
-
-cluster: {
- metadata: {
- name: string
- }
-}
-component: {
- clusterDefName: string
- name: string
-}
-backupPolicyName: string
-backupJobKey: {
- Name: string
- Namespace: string
-}
-backupType: string
-backupJob: {
- apiVersion: "dataprotection.kubeblocks.io/v1alpha1"
- kind: "Backup"
- metadata: {
- name: backupJobKey.Name
- namespace: backupJobKey.Namespace
- labels: {
- "dataprotection.kubeblocks.io/backup-type": backupType
- "apps.kubeblocks.io/managed-by": "cluster"
- "backuppolicies.dataprotection.kubeblocks.io/name": backupPolicyName
- "app.kubernetes.io/name": "\(component.clusterDefName)"
- "app.kubernetes.io/instance": cluster.metadata.name
- "app.kubernetes.io/managed-by": "kubeblocks"
- "apps.kubeblocks.io/component-name": "\(component.name)"
- }
- }
- spec: {
- "backupPolicyName": backupPolicyName
- "backupType": backupType
- }
-}
diff --git a/internal/controller/factory/cue/backup_manifests_template.cue b/internal/controller/factory/cue/backup_manifests_template.cue
deleted file mode 100644
index cd41196c069..00000000000
--- a/internal/controller/factory/cue/backup_manifests_template.cue
+++ /dev/null
@@ -1,50 +0,0 @@
-// Copyright (C) 2022-2023 ApeCloud Co., Ltd
-//
-// This file is part of KubeBlocks project
-//
-// This program is free software: you can redistribute it and/or modify
-// it under the terms of the GNU Affero General Public License as published by
-// the Free Software Foundation, either version 3 of the License, or
-// (at your option) any later version.
-//
-// This program is distributed in the hope that it will be useful
-// but WITHOUT ANY WARRANTY; without even the implied warranty of
-// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
-// GNU Affero General Public License for more details.
-//
-// You should have received a copy of the GNU Affero General Public License
-// along with this program. If not, see .
-
-backup: {
- metadata: {
- name: string
- namespace: string
- }
-}
-
-podSpec: {
- containers: [...]
- volumes: [...]
- env: [...]
- restartPolicy: "Never"
- securityContext:
- runAsUser: 0
-}
-
-job: {
- apiVersion: "batch/v1"
- kind: "Job"
- metadata: {
- name: string
- namespace: string
- labels:
- "app.kubernetes.io/managed-by": "kubeblocks"
- }
- spec: {
- template: {
- spec: podSpec
- }
- backOffLimit: 3
- ttlSecondsAfterFinished: 10
- }
-}
diff --git a/internal/controller/factory/cue/config_manager_sidecar.cue b/internal/controller/factory/cue/config_manager_sidecar.cue
deleted file mode 100644
index 9eeea7dd637..00000000000
--- a/internal/controller/factory/cue/config_manager_sidecar.cue
+++ /dev/null
@@ -1,107 +0,0 @@
-// Copyright (C) 2022-2023 ApeCloud Co., Ltd
-//
-// This file is part of KubeBlocks project
-//
-// This program is free software: you can redistribute it and/or modify
-// it under the terms of the GNU Affero General Public License as published by
-// the Free Software Foundation, either version 3 of the License, or
-// (at your option) any later version.
-//
-// This program is distributed in the hope that it will be useful
-// but WITHOUT ANY WARRANTY; without even the implied warranty of
-// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
-// GNU Affero General Public License for more details.
-//
-// You should have received a copy of the GNU Affero General Public License
-// along with this program. If not, see .
-
-template: {
- name: parameter.name
- command: [
- "env",
- ]
- args: [
- "PATH=/usr/local/sbin:/usr/local/bin:/usr/sbin:/usr/bin:/sbin:/bin:$(TOOLS_PATH)",
- "/bin/reloader",
- for arg in parameter.args {
- arg
- },
- ]
- env: [
- {
- name: "CONFIG_MANAGER_POD_IP"
- valueFrom: {
- fieldRef: {
- apiVersion: "v1"
- fieldPath: "status.podIP"
- }
- }
- },
- if parameter.characterType != "" {
- {
- name: "DB_TYPE"
- value: parameter.characterType
- }
- },
- if parameter.characterType == "mysql" {
- {
- name: "MYSQL_USER"
- valueFrom: {
- secretKeyRef: {
- key: "username"
- name: parameter.secreteName
- }
- }
- }
- },
- if parameter.characterType == "mysql" {
- {
- name: "MYSQL_PASSWORD"
- valueFrom: {
- secretKeyRef: {
- key: "password"
- name: parameter.secreteName
- }
- }
- }
- },
- if parameter.characterType == "mysql" {
- {
- name: "DATA_SOURCE_NAME"
- value: "$(MYSQL_USER):$(MYSQL_PASSWORD)@(localhost:3306)/"
- }
- },
- // other type
- ]
-
- image: parameter.sidecarImage
- imagePullPolicy: "IfNotPresent"
- volumeMounts: parameter.volumes
- if parameter.shareProcessNamespace {
- {
- securityContext:
- runAsUser: 0
- defaultAllowPrivilegeEscalation: false
- }
- }
-}
-
-#ArgType: string
-#EnvType: {
- name: string
- value: string
-
- // valueFrom
- ...
-}
-
-parameter: {
- name: string
- characterType: string
- sidecarImage: string
- secreteName: string
- shareProcessNamespace: bool
- args: [...#ArgType]
- // envs?: [...#EnvType]
- volumes: [...]
-}
diff --git a/internal/controller/factory/cue/config_template.cue b/internal/controller/factory/cue/config_template.cue
deleted file mode 100644
index f07dad738bf..00000000000
--- a/internal/controller/factory/cue/config_template.cue
+++ /dev/null
@@ -1,64 +0,0 @@
-// Copyright (C) 2022-2023 ApeCloud Co., Ltd
-//
-// This file is part of KubeBlocks project
-//
-// This program is free software: you can redistribute it and/or modify
-// it under the terms of the GNU Affero General Public License as published by
-// the Free Software Foundation, either version 3 of the License, or
-// (at your option) any later version.
-//
-// This program is distributed in the hope that it will be useful
-// but WITHOUT ANY WARRANTY; without even the implied warranty of
-// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
-// GNU Affero General Public License for more details.
-//
-// You should have received a copy of the GNU Affero General Public License
-// along with this program. If not, see .
-
-meta: {
- clusterDefinition: {
- name: string
- }
-
- cluster: {
- namespace: string
- name: string
- }
-
- component: {
- name: string
- configName: string
- templateName: string
- configConstraintsName: string
- configTemplateName: string
- compDefName: string
- }
-}
-
-config: {
- apiVersion: "v1"
- kind: "ConfigMap"
- metadata: {
- name: meta.component.configName
- namespace: meta.cluster.namespace
- labels: {
- "app.kubernetes.io/name": "\(meta.clusterDefinition.name)"
- "app.kubernetes.io/instance": meta.cluster.name
- "app.kubernetes.io/managed-by": "kubeblocks"
- "app.kubernetes.io/component": "\(meta.component.compDefName)"
-
- "apps.kubeblocks.io/component-name": "\(meta.component.name)"
- // configmap selector for ConfigureController
- "config.kubeblocks.io/config-type": "instance"
- // config template name
- "config.kubeblocks.io/template-name": "\(meta.component.templateName)"
- }
- annotations: {
- // enable configmap upgrade
- "config.kubeblocks.io/disable-reconfigure": "false"
- }
-
- data: {
- }
- }
-}
diff --git a/internal/controller/factory/cue/conn_credential_template.cue b/internal/controller/factory/cue/conn_credential_template.cue
deleted file mode 100644
index 237012a497e..00000000000
--- a/internal/controller/factory/cue/conn_credential_template.cue
+++ /dev/null
@@ -1,49 +0,0 @@
-// Copyright (C) 2022-2023 ApeCloud Co., Ltd
-//
-// This file is part of KubeBlocks project
-//
-// This program is free software: you can redistribute it and/or modify
-// it under the terms of the GNU Affero General Public License as published by
-// the Free Software Foundation, either version 3 of the License, or
-// (at your option) any later version.
-//
-// This program is distributed in the hope that it will be useful
-// but WITHOUT ANY WARRANTY; without even the implied warranty of
-// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
-// GNU Affero General Public License for more details.
-//
-// You should have received a copy of the GNU Affero General Public License
-// along with this program. If not, see .
-
-clusterdefinition: {
- metadata: {
- name: string
- }
- spec: {
- type: string
- connectionCredential: {...}
- }
-}
-cluster: {
- metadata: {
- namespace: string
- name: string
- }
-}
-secret: {
- apiVersion: "v1"
- stringData: clusterdefinition.spec.connectionCredential
- kind: "Secret"
- metadata: {
- name: "\(cluster.metadata.name)-conn-credential"
- namespace: cluster.metadata.namespace
- labels: {
- "app.kubernetes.io/name": "\(clusterdefinition.metadata.name)"
- "app.kubernetes.io/instance": cluster.metadata.name
- "app.kubernetes.io/managed-by": "kubeblocks"
- if clusterdefinition.spec.type != _|_ {
- "apps.kubeblocks.io/cluster-type": clusterdefinition.spec.type
- }
- }
- }
-}
diff --git a/internal/controller/factory/cue/deployment_template.cue b/internal/controller/factory/cue/deployment_template.cue
deleted file mode 100644
index c58870d9dc4..00000000000
--- a/internal/controller/factory/cue/deployment_template.cue
+++ /dev/null
@@ -1,84 +0,0 @@
-// Copyright (C) 2022-2023 ApeCloud Co., Ltd
-//
-// This file is part of KubeBlocks project
-//
-// This program is free software: you can redistribute it and/or modify
-// it under the terms of the GNU Affero General Public License as published by
-// the Free Software Foundation, either version 3 of the License, or
-// (at your option) any later version.
-//
-// This program is distributed in the hope that it will be useful
-// but WITHOUT ANY WARRANTY; without even the implied warranty of
-// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
-// GNU Affero General Public License for more details.
-//
-// You should have received a copy of the GNU Affero General Public License
-// along with this program. If not, see .
-
-cluster: {
- metadata: {
- namespace: string
- name: string
- }
- spec: {
- clusterVersionRef: string
- }
-}
-component: {
- clusterDefName: string
- name: string
- compDefName: string
- workloadType: string
- replicas: int
- podSpec: {
- containers: [...]
- enableServiceLinks: bool | *false
- }
- volumeClaimTemplates: [...]
-}
-
-deployment: {
- "apiVersion": "apps/v1"
- "kind": "Deployment"
- "metadata": {
- namespace: cluster.metadata.namespace
- name: "\(cluster.metadata.name)-\(component.name)"
- labels: {
- "app.kubernetes.io/name": "\(component.clusterDefName)"
- "app.kubernetes.io/instance": cluster.metadata.name
- "app.kubernetes.io/managed-by": "kubeblocks"
- "app.kubernetes.io/component": "\(component.compDefName)"
-
- "apps.kubeblocks.io/component-name": "\(component.name)"
- }
- }
- "spec": {
- replicas: component.replicas
- minReadySeconds: 10
- selector: {
- matchLabels: {
- "app.kubernetes.io/name": "\(component.clusterDefName)"
- "app.kubernetes.io/instance": "\(cluster.metadata.name)"
- "app.kubernetes.io/managed-by": "kubeblocks"
-
- "apps.kubeblocks.io/component-name": "\(component.name)"
- }
- }
- template: {
- metadata: {
- labels: {
- "app.kubernetes.io/name": "\(component.clusterDefName)"
- "app.kubernetes.io/instance": "\(cluster.metadata.name)"
- "app.kubernetes.io/managed-by": "kubeblocks"
- "app.kubernetes.io/component": "\(component.compDefName)"
- if cluster.spec.clusterVersionRef != _|_ {
- "app.kubernetes.io/version": "\(cluster.spec.clusterVersionRef)"
- }
- "apps.kubeblocks.io/component-name": "\(component.name)"
- "apps.kubeblocks.io/workload-type": "\(component.workloadType)"
- }
- }
- spec: component.podSpec
- }
- }
-}
diff --git a/internal/controller/factory/cue/env_config_template.cue b/internal/controller/factory/cue/env_config_template.cue
deleted file mode 100644
index f2babc484c3..00000000000
--- a/internal/controller/factory/cue/env_config_template.cue
+++ /dev/null
@@ -1,50 +0,0 @@
-// Copyright (C) 2022-2023 ApeCloud Co., Ltd
-//
-// This file is part of KubeBlocks project
-//
-// This program is free software: you can redistribute it and/or modify
-// it under the terms of the GNU Affero General Public License as published by
-// the Free Software Foundation, either version 3 of the License, or
-// (at your option) any later version.
-//
-// This program is distributed in the hope that it will be useful
-// but WITHOUT ANY WARRANTY; without even the implied warranty of
-// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
-// GNU Affero General Public License for more details.
-//
-// You should have received a copy of the GNU Affero General Public License
-// along with this program. If not, see .
-
-cluster: {
- metadata: {
- namespace: string
- name: string
- }
-}
-component: {
- name: string
- clusterDefName: string
- compDefName: string
-}
-
-config: {
- apiVersion: "v1"
- kind: "ConfigMap"
- metadata: {
- // this naming pattern has been referenced elsewhere, complete code scan is
- // required if this naming pattern is going be changed.
- name: "\(cluster.metadata.name)-\(component.name)-env"
- namespace: cluster.metadata.namespace
- labels: {
- "app.kubernetes.io/name": "\(component.clusterDefName)"
- "app.kubernetes.io/instance": cluster.metadata.name
- "app.kubernetes.io/managed-by": "kubeblocks"
- "app.kubernetes.io/component": "\(component.compDefName)"
-
- // configmap selector for env update
- "apps.kubeblocks.io/config-type": "kubeblocks-env"
- "apps.kubeblocks.io/component-name": "\(component.name)"
- }
- }
- data: [string]: string
-}
diff --git a/internal/controller/factory/cue/headless_service_template.cue b/internal/controller/factory/cue/headless_service_template.cue
deleted file mode 100644
index c378fce32eb..00000000000
--- a/internal/controller/factory/cue/headless_service_template.cue
+++ /dev/null
@@ -1,88 +0,0 @@
-// Copyright (C) 2022-2023 ApeCloud Co., Ltd
-//
-// This file is part of KubeBlocks project
-//
-// This program is free software: you can redistribute it and/or modify
-// it under the terms of the GNU Affero General Public License as published by
-// the Free Software Foundation, either version 3 of the License, or
-// (at your option) any later version.
-//
-// This program is distributed in the hope that it will be useful
-// but WITHOUT ANY WARRANTY; without even the implied warranty of
-// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
-// GNU Affero General Public License for more details.
-//
-// You should have received a copy of the GNU Affero General Public License
-// along with this program. If not, see .
-
-cluster: {
- metadata: {
- namespace: string
- name: string
- }
-}
-component: {
- clusterDefName: string
- compDefName: string
- name: string
- monitor: {
- enable: bool
- builtIn: bool
- scrapePort: int
- scrapePath: string
- }
- podSpec: containers: [...]
-}
-
-service: {
- "apiVersion": "v1"
- "kind": "Service"
- "metadata": {
- namespace: cluster.metadata.namespace
- name: "\(cluster.metadata.name)-\(component.name)-headless"
- labels: {
- "app.kubernetes.io/name": "\(component.clusterDefName)"
- "app.kubernetes.io/instance": cluster.metadata.name
- "app.kubernetes.io/managed-by": "kubeblocks"
- "app.kubernetes.io/component": "\(component.compDefName)"
-
- "apps.kubeblocks.io/component-name": "\(component.name)"
- }
- annotations: {
- if component.monitor.enable == false {
- "monitor.kubeblocks.io/scrape": "false"
- "monitor.kubeblocks.io/agamotto": "false"
- }
- if component.monitor.enable == true && component.monitor.builtIn == false {
- "monitor.kubeblocks.io/scrape": "true"
- "monitor.kubeblocks.io/path": component.monitor.scrapePath
- "monitor.kubeblocks.io/port": "\(component.monitor.scrapePort)"
- "monitor.kubeblocks.io/scheme": "http"
- "monitor.kubeblocks.io/agamotto": "false"
- }
- if component.monitor.enable == true && component.monitor.builtIn == true {
- "monitor.kubeblocks.io/scrape": "false"
- "monitor.kubeblocks.io/agamotto": "true"
- }
- }
- }
- "spec": {
- "type": "ClusterIP"
- "clusterIP": "None"
- "selector": {
- "app.kubernetes.io/instance": "\(cluster.metadata.name)"
- "app.kubernetes.io/managed-by": "kubeblocks"
-
- "apps.kubeblocks.io/component-name": "\(component.name)"
- }
- ports: [
- for _, container in component.podSpec.containers if container.ports != _|_
- for _, v in container.ports {
- name: v.name
- protocol: v.protocol
- port: v.containerPort
- targetPort: v.name
- },
- ]
- }
-}
diff --git a/internal/controller/factory/cue/pdb_template.cue b/internal/controller/factory/cue/pdb_template.cue
deleted file mode 100644
index 1c89fa6f3f5..00000000000
--- a/internal/controller/factory/cue/pdb_template.cue
+++ /dev/null
@@ -1,60 +0,0 @@
-// Copyright (C) 2022-2023 ApeCloud Co., Ltd
-//
-// This file is part of KubeBlocks project
-//
-// This program is free software: you can redistribute it and/or modify
-// it under the terms of the GNU Affero General Public License as published by
-// the Free Software Foundation, either version 3 of the License, or
-// (at your option) any later version.
-//
-// This program is distributed in the hope that it will be useful
-// but WITHOUT ANY WARRANTY; without even the implied warranty of
-// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
-// GNU Affero General Public License for more details.
-//
-// You should have received a copy of the GNU Affero General Public License
-// along with this program. If not, see .
-
-cluster: {
- metadata: {
- namespace: string
- name: string
- }
-}
-component: {
- clusterDefName: string
- compDefName: string
- name: string
- minAvailable: string | int
-}
-
-pdb: {
- "apiVersion": "policy/v1"
- "kind": "PodDisruptionBudget"
- "metadata": {
- namespace: cluster.metadata.namespace
- name: "\(cluster.metadata.name)-\(component.name)"
- labels: {
- "app.kubernetes.io/name": "\(component.clusterDefName)"
- "app.kubernetes.io/instance": cluster.metadata.name
- "app.kubernetes.io/managed-by": "kubeblocks"
- "app.kubernetes.io/component": "\(component.compDefName)"
-
- "apps.kubeblocks.io/component-name": "\(component.name)"
- }
- }
- "spec": {
- if component.minAvailable != _|_ {
- minAvailable: component.minAvailable
- }
- selector: {
- matchLabels: {
- "app.kubernetes.io/name": "\(component.clusterDefName)"
- "app.kubernetes.io/instance": "\(cluster.metadata.name)"
- "app.kubernetes.io/managed-by": "kubeblocks"
-
- "apps.kubeblocks.io/component-name": "\(component.name)"
- }
- }
- }
-}
diff --git a/internal/controller/factory/cue/pvc_template.cue b/internal/controller/factory/cue/pvc_template.cue
deleted file mode 100644
index 1eb451f3c71..00000000000
--- a/internal/controller/factory/cue/pvc_template.cue
+++ /dev/null
@@ -1,73 +0,0 @@
-// Copyright (C) 2022-2023 ApeCloud Co., Ltd
-//
-// This file is part of KubeBlocks project
-//
-// This program is free software: you can redistribute it and/or modify
-// it under the terms of the GNU Affero General Public License as published by
-// the Free Software Foundation, either version 3 of the License, or
-// (at your option) any later version.
-//
-// This program is distributed in the hope that it will be useful
-// but WITHOUT ANY WARRANTY; without even the implied warranty of
-// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
-// GNU Affero General Public License for more details.
-//
-// You should have received a copy of the GNU Affero General Public License
-// along with this program. If not, see .
-
-cluster: {
- metadata: {
- name: string
- }
-}
-component: {
- clusterDefName: string
- name: string
-}
-volumeClaimTemplate: {
- metadata: {
- name: string
- }
- spec: {
- accessModes: [string]
- resources: {}
- }
-}
-snapshot_name: string
-pvc_key: {
- Name: string
- Namespace: string
-}
-pvc: {
- kind: "PersistentVolumeClaim"
- apiVersion: "v1"
- metadata: {
- name: pvc_key.Name
- namespace: pvc_key.Namespace
- labels: {
- "apps.kubeblocks.io/vct-name": volumeClaimTemplate.metadata.name
- if component.clusterDefName != _|_ {
- "app.kubernetes.io/name": "\(component.clusterDefName)"
- }
- if component.name != _|_ {
- "apps.kubeblocks.io/component-name": "\(component.name)"
- }
- "app.kubernetes.io/instance": cluster.metadata.name
- "app.kubernetes.io/managed-by": "kubeblocks"
- }
- }
- spec: {
- accessModes: volumeClaimTemplate.spec.accessModes
- resources: volumeClaimTemplate.spec.resources
- if volumeClaimTemplate.spec.storageClassName != _|_ {
- storageClassName: volumeClaimTemplate.spec.storageClassName
- }
- if len(snapshot_name) > 0 {
- dataSource: {
- "name": snapshot_name
- "kind": "VolumeSnapshot"
- "apiGroup": "snapshot.storage.k8s.io"
- }
- }
- }
-}
diff --git a/internal/controller/factory/cue/rbac_template.cue b/internal/controller/factory/cue/rbac_template.cue
deleted file mode 100644
index d3c75799d7f..00000000000
--- a/internal/controller/factory/cue/rbac_template.cue
+++ /dev/null
@@ -1,87 +0,0 @@
-// Copyright (C) 2022-2023 ApeCloud Co., Ltd
-//
-// This file is part of KubeBlocks project
-//
-// This program is free software: you can redistribute it and/or modify
-// it under the terms of the GNU Affero General Public License as published by
-// the Free Software Foundation, either version 3 of the License, or
-// (at your option) any later version.
-//
-// This program is distributed in the hope that it will be useful
-// but WITHOUT ANY WARRANTY; without even the implied warranty of
-// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
-// GNU Affero General Public License for more details.
-//
-// You should have received a copy of the GNU Affero General Public License
-// along with this program. If not, see .
-
-cluster: {
- metadata: {
- namespace: string
- name: string
- }
- spec: {
- clusterDefinitionRef: string
- }
-}
-
-serviceaccount: {
- apiVersion: "v1"
- kind: "ServiceAccount"
- metadata: {
- namespace: cluster.metadata.namespace
- name: "kb-\(cluster.metadata.name)"
- labels: {
- "app.kubernetes.io/name": cluster.spec.clusterDefinitionRef
- "app.kubernetes.io/instance": cluster.metadata.name
- "app.kubernetes.io/managed-by": "kubeblocks"
- }
- }
-}
-
-rolebinding: {
- apiVersion: "rbac.authorization.k8s.io/v1"
- kind: "RoleBinding"
- metadata: {
- name: "kb-\(cluster.metadata.name)"
- namespace: cluster.metadata.namespace
- labels: {
- "app.kubernetes.io/name": cluster.spec.clusterDefinitionRef
- "app.kubernetes.io/instance": cluster.metadata.name
- "app.kubernetes.io/managed-by": "kubeblocks"
- }
- }
- roleRef: {
- apiGroup: "rbac.authorization.k8s.io"
- kind: "ClusterRole"
- name: "kubeblocks-cluster-pod-role"
- }
- subjects: [{
- kind: "ServiceAccount"
- name: "kb-\(cluster.metadata.name)"
- namespace: cluster.metadata.namespace
- }]
-}
-
-clusterrolebinding: {
- apiVersion: "rbac.authorization.k8s.io/v1"
- kind: "ClusterRoleBinding"
- metadata: {
- name: "kb-\(cluster.metadata.name)"
- labels: {
- "app.kubernetes.io/name": cluster.spec.clusterDefinitionRef
- "app.kubernetes.io/instance": cluster.metadata.name
- "app.kubernetes.io/managed-by": "kubeblocks"
- }
- }
- roleRef: {
- apiGroup: "rbac.authorization.k8s.io"
- kind: "ClusterRole"
- name: "kubeblocks-volume-protection-pod-role"
- }
- subjects: [{
- kind: "ServiceAccount"
- name: "kb-\(cluster.metadata.name)"
- namespace: cluster.metadata.namespace
- }]
-}
diff --git a/internal/controller/factory/cue/restore_job_template.cue b/internal/controller/factory/cue/restore_job_template.cue
deleted file mode 100644
index 50c65a5ce87..00000000000
--- a/internal/controller/factory/cue/restore_job_template.cue
+++ /dev/null
@@ -1,49 +0,0 @@
-// Copyright (C) 2022-2023 ApeCloud Co., Ltd
-//
-// This file is part of KubeBlocks project
-//
-// This program is free software: you can redistribute it and/or modify
-// it under the terms of the GNU Affero General Public License as published by
-// the Free Software Foundation, either version 3 of the License, or
-// (at your option) any later version.
-//
-// This program is distributed in the hope that it will be useful
-// but WITHOUT ANY WARRANTY; without even the implied warranty of
-// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
-// GNU Affero General Public License for more details.
-//
-// You should have received a copy of the GNU Affero General Public License
-// along with this program. If not, see .
-
-container: {
- name: "restore"
- image: string
- imagePullPolicy: "IfNotPresent"
- command: [...]
- volumeMounts: [...]
- env: [...]
- resources: {}
-}
-
-job: {
- apiVersion: "batch/v1"
- kind: "Job"
- metadata: {
- name: string
- namespace: string
- labels: {
- "app.kubernetes.io/managed-by": "kubeblocks"
- }
- }
- spec: {
- template: {
- spec: {
- containers: [container]
- volumes: [...]
- restartPolicy: "OnFailure"
- securityContext:
- runAsUser: 0
- }
- }
- }
-}
diff --git a/internal/controller/factory/cue/service_template.cue b/internal/controller/factory/cue/service_template.cue
deleted file mode 100644
index 1359763b1d6..00000000000
--- a/internal/controller/factory/cue/service_template.cue
+++ /dev/null
@@ -1,81 +0,0 @@
-// Copyright (C) 2022-2023 ApeCloud Co., Ltd
-//
-// This file is part of KubeBlocks project
-//
-// This program is free software: you can redistribute it and/or modify
-// it under the terms of the GNU Affero General Public License as published by
-// the Free Software Foundation, either version 3 of the License, or
-// (at your option) any later version.
-//
-// This program is distributed in the hope that it will be useful
-// but WITHOUT ANY WARRANTY; without even the implied warranty of
-// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
-// GNU Affero General Public License for more details.
-//
-// You should have received a copy of the GNU Affero General Public License
-// along with this program. If not, see .
-
-cluster: {
- metadata: {
- namespace: string
- name: string
- }
-}
-
-component: {
- clusterDefName: string
- compDefName: string
- name: string
-}
-
-service: {
- metadata: {
- name: string
- annotations: {}
- }
- spec: {
- ports: [...]
- type: string
- }
-}
-
-svc: {
- "apiVersion": "v1"
- "kind": "Service"
- "metadata": {
- namespace: cluster.metadata.namespace
- if service.metadata.name != _|_ {
- name: "\(cluster.metadata.name)-\(component.name)-\(service.metadata.name)"
- }
- if service.metadata.name == _|_ {
- name: "\(cluster.metadata.name)-\(component.name)"
- }
- labels: {
- "app.kubernetes.io/name": "\(component.clusterDefName)"
- "app.kubernetes.io/instance": cluster.metadata.name
- "app.kubernetes.io/managed-by": "kubeblocks"
- "app.kubernetes.io/component": "\(component.compDefName)"
-
- "apps.kubeblocks.io/component-name": "\(component.name)"
- }
- annotations: service.metadata.annotations
- }
- "spec": {
- "selector": {
- "app.kubernetes.io/instance": "\(cluster.metadata.name)"
- "app.kubernetes.io/managed-by": "kubeblocks"
-
- "apps.kubeblocks.io/component-name": "\(component.name)"
- }
- ports: service.spec.ports
- if service.spec.type != _|_ {
- type: service.spec.type
- }
- if service.spec.type == "LoadBalancer" {
- // Set externalTrafficPolicy to Local has two benefits:
- // 1. preserve client IP
- // 2. improve network performance by reducing one hop
- externalTrafficPolicy: "Local"
- }
- }
-}
diff --git a/internal/controller/factory/cue/statefulset_template.cue b/internal/controller/factory/cue/statefulset_template.cue
deleted file mode 100644
index fa825577aeb..00000000000
--- a/internal/controller/factory/cue/statefulset_template.cue
+++ /dev/null
@@ -1,85 +0,0 @@
-// Copyright (C) 2022-2023 ApeCloud Co., Ltd
-//
-// This file is part of KubeBlocks project
-//
-// This program is free software: you can redistribute it and/or modify
-// it under the terms of the GNU Affero General Public License as published by
-// the Free Software Foundation, either version 3 of the License, or
-// (at your option) any later version.
-//
-// This program is distributed in the hope that it will be useful
-// but WITHOUT ANY WARRANTY; without even the implied warranty of
-// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
-// GNU Affero General Public License for more details.
-//
-// You should have received a copy of the GNU Affero General Public License
-// along with this program. If not, see .
-
-cluster: {
- metadata: {
- namespace: string
- name: string
- }
- spec: {
- clusterVersionRef: string
- }
-}
-component: {
- clusterDefName: string
- compDefName: string
- name: string
- workloadType: string
- replicas: int
- podSpec: {
- containers: [...]
- enableServiceLinks: bool | *false
- }
- volumeClaimTemplates: [...]
-}
-
-statefulset: {
- apiVersion: "apps/v1"
- kind: "StatefulSet"
- metadata: {
- namespace: cluster.metadata.namespace
- name: "\(cluster.metadata.name)-\(component.name)"
- labels: {
- "app.kubernetes.io/name": "\(component.clusterDefName)"
- "app.kubernetes.io/instance": cluster.metadata.name
- "app.kubernetes.io/managed-by": "kubeblocks"
- "app.kubernetes.io/component": "\(component.compDefName)"
-
- "apps.kubeblocks.io/component-name": "\(component.name)"
- }
- }
- spec: {
- selector:
- matchLabels: {
- "app.kubernetes.io/name": "\(component.clusterDefName)"
- "app.kubernetes.io/instance": "\(cluster.metadata.name)"
- "app.kubernetes.io/managed-by": "kubeblocks"
-
- "apps.kubeblocks.io/component-name": "\(component.name)"
- }
- serviceName: "\(cluster.metadata.name)-\(component.name)-headless"
- replicas: component.replicas
- template: {
- metadata: {
- labels: {
- "app.kubernetes.io/name": "\(component.clusterDefName)"
- "app.kubernetes.io/instance": "\(cluster.metadata.name)"
- "app.kubernetes.io/managed-by": "kubeblocks"
- "app.kubernetes.io/component": "\(component.compDefName)"
- if cluster.spec.clusterVersionRef != _|_ {
- "app.kubernetes.io/version": "\(cluster.spec.clusterVersionRef)"
- }
-
- "apps.kubeblocks.io/component-name": "\(component.name)"
- "apps.kubeblocks.io/workload-type": "\(component.workloadType)"
- }
- }
- spec: component.podSpec
- }
- volumeClaimTemplates: component.volumeClaimTemplates
- }
-}
diff --git a/internal/controller/factory/cue/volumesnapshotclass.cue b/internal/controller/factory/cue/volumesnapshotclass.cue
deleted file mode 100644
index 9759ae77581..00000000000
--- a/internal/controller/factory/cue/volumesnapshotclass.cue
+++ /dev/null
@@ -1,29 +0,0 @@
-// Copyright (C) 2022-2023 ApeCloud Co., Ltd
-//
-// This file is part of KubeBlocks project
-//
-// This program is free software: you can redistribute it and/or modify
-// it under the terms of the GNU Affero General Public License as published by
-// the Free Software Foundation, either version 3 of the License, or
-// (at your option) any later version.
-//
-// This program is distributed in the hope that it will be useful
-// but WITHOUT ANY WARRANTY; without even the implied warranty of
-// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
-// GNU Affero General Public License for more details.
-//
-// You should have received a copy of the GNU Affero General Public License
-// along with this program. If not, see .
-
-class: {
- apiVersion: "snapshot.storage.k8s.io/v1"
- kind: "VolumeSnapshotClass"
- metadata: {
- name: string
- labels: {
- "app.kubernetes.io/managed-by": "kubeblocks"
- }
- }
- driver: string
- deletionPolicy: "Delete"
-}
diff --git a/internal/controller/factory/suite_test.go b/internal/controller/factory/suite_test.go
index a11f020e4f8..f599feedbe8 100644
--- a/internal/controller/factory/suite_test.go
+++ b/internal/controller/factory/suite_test.go
@@ -41,7 +41,7 @@ import (
"sigs.k8s.io/controller-runtime/pkg/log/zap"
appsv1alpha1 "github.com/apecloud/kubeblocks/apis/apps/v1alpha1"
- dataprotectionv1alpha1 "github.com/apecloud/kubeblocks/apis/dataprotection/v1alpha1"
+ dpv1alpha1 "github.com/apecloud/kubeblocks/apis/dataprotection/v1alpha1"
intctrlutil "github.com/apecloud/kubeblocks/internal/controllerutil"
"github.com/apecloud/kubeblocks/internal/testutil"
viper "github.com/apecloud/kubeblocks/internal/viperx"
@@ -100,7 +100,7 @@ var _ = BeforeSuite(func() {
err = appsv1alpha1.AddToScheme(scheme.Scheme)
Expect(err).NotTo(HaveOccurred())
- err = dataprotectionv1alpha1.AddToScheme(scheme.Scheme)
+ err = dpv1alpha1.AddToScheme(scheme.Scheme)
Expect(err).NotTo(HaveOccurred())
err = snapshotv1.AddToScheme(scheme.Scheme)
diff --git a/internal/controller/handler/handler_builder.go b/internal/controller/handler/handler_builder.go
index 947b145ef63..5cb345fb3fe 100644
--- a/internal/controller/handler/handler_builder.go
+++ b/internal/controller/handler/handler_builder.go
@@ -20,6 +20,8 @@ along with this program. If not, see .
package handler
import (
+ "context"
+
"sigs.k8s.io/controller-runtime/pkg/client"
"sigs.k8s.io/controller-runtime/pkg/handler"
"sigs.k8s.io/controller-runtime/pkg/reconcile"
@@ -50,7 +52,7 @@ func (builder *realBuilder) AddFinder(finder Finder) Builder {
}
func (builder *realBuilder) Build() handler.EventHandler {
- fn := func(obj client.Object) []reconcile.Request {
+ fn := func(ctx context.Context, obj client.Object) []reconcile.Request {
var key *model.GVKNObjKey
for i, finder := range builder.finders {
key = finder.Find(builder.ctx, obj)
diff --git a/internal/controller/handler/handler_builder_test.go b/internal/controller/handler/handler_builder_test.go
index 424f99ad115..03d900d053d 100644
--- a/internal/controller/handler/handler_builder_test.go
+++ b/internal/controller/handler/handler_builder_test.go
@@ -104,22 +104,22 @@ var _ = Describe("handler builder test.", func() {
}{
{
name: "Create",
- testFunc: func() { handler.Create(createEvent, queue) },
+ testFunc: func() { handler.Create(ctx, createEvent, queue) },
getTimes: 1,
},
{
name: "Update",
- testFunc: func() { handler.Update(updateEvent, queue) },
+ testFunc: func() { handler.Update(ctx, updateEvent, queue) },
getTimes: 2,
},
{
name: "Delete",
- testFunc: func() { handler.Delete(deleteEvent, queue) },
+ testFunc: func() { handler.Delete(ctx, deleteEvent, queue) },
getTimes: 1,
},
{
name: "Generic",
- testFunc: func() { handler.Generic(genericEvent, queue) },
+ testFunc: func() { handler.Generic(ctx, genericEvent, queue) },
getTimes: 1,
},
}
diff --git a/internal/controller/plan/prepare.go b/internal/controller/plan/prepare.go
index 5dc97857b0b..3dd42bfa057 100644
--- a/internal/controller/plan/prepare.go
+++ b/internal/controller/plan/prepare.go
@@ -20,10 +20,11 @@ along with this program. If not, see .
package plan
import (
- intctrlutil "github.com/apecloud/kubeblocks/internal/controllerutil"
corev1 "k8s.io/api/core/v1"
"sigs.k8s.io/controller-runtime/pkg/client"
+ intctrlutil "github.com/apecloud/kubeblocks/internal/controllerutil"
+
appsv1alpha1 "github.com/apecloud/kubeblocks/apis/apps/v1alpha1"
"github.com/apecloud/kubeblocks/internal/controller/component"
"github.com/apecloud/kubeblocks/internal/controller/configuration"
diff --git a/internal/controller/plan/prepare_test.go b/internal/controller/plan/prepare_test.go
index a859c708835..6cea96e52b8 100644
--- a/internal/controller/plan/prepare_test.go
+++ b/internal/controller/plan/prepare_test.go
@@ -33,7 +33,6 @@ import (
appsv1alpha1 "github.com/apecloud/kubeblocks/apis/apps/v1alpha1"
cfgcore "github.com/apecloud/kubeblocks/internal/configuration/core"
- "github.com/apecloud/kubeblocks/internal/constant"
"github.com/apecloud/kubeblocks/internal/controller/component"
"github.com/apecloud/kubeblocks/internal/controller/configuration"
"github.com/apecloud/kubeblocks/internal/controller/factory"
@@ -63,10 +62,7 @@ func buildComponentResources(reqCtx intctrlutil.RequestCtx, cli client.Client,
cluster.UID = types.UID("test-uid")
}
workloadProcessor := func(customSetup func(*corev1.ConfigMap) (client.Object, error)) error {
- envConfig, err := factory.BuildEnvConfig(cluster, component)
- if err != nil {
- return err
- }
+ envConfig := factory.BuildEnvConfig(cluster, component)
resources = append(resources, envConfig)
workload, err := customSetup(envConfig)
@@ -79,12 +75,6 @@ func buildComponentResources(reqCtx intctrlutil.RequestCtx, cli client.Client,
resources = append(resources, workload)
}()
- svc, err := factory.BuildHeadlessSvc(cluster, component)
- if err != nil {
- return err
- }
- resources = append(resources, svc)
-
var podSpec *corev1.PodSpec
sts, ok := workload.(*appsv1.StatefulSet)
if ok {
@@ -138,41 +128,16 @@ func buildComponentResources(reqCtx intctrlutil.RequestCtx, cli client.Client,
// if no these handle, the cluster controller will occur an error during reconciling.
// conditional build PodDisruptionBudget
if component.MinAvailable != nil {
- pdb, err := factory.BuildPDB(cluster, component)
- if err != nil {
- return nil, err
- }
+ pdb := factory.BuildPDB(cluster, component)
resources = append(resources, pdb)
} else {
panic("this shouldn't happen")
}
- svcList, err := factory.BuildSvcListWithCustomAttributes(cluster, component, func(svc *corev1.Service) {
- switch component.WorkloadType {
- case appsv1alpha1.Consensus:
- addLeaderSelectorLabels(svc, component)
- case appsv1alpha1.Replication:
- svc.Spec.Selector[constant.RoleLabelKey] = "primary"
- }
- })
- if err != nil {
- return nil, err
- }
- for _, svc := range svcList {
- resources = append(resources, svc)
- }
-
// REVIEW/TODO:
// - need higher level abstraction handling
// - or move this module to part operator controller handling
switch component.WorkloadType {
- case appsv1alpha1.Stateless:
- if err := workloadProcessor(
- func(envConfig *corev1.ConfigMap) (client.Object, error) {
- return factory.BuildDeploy(reqCtx, cluster, component, "")
- }); err != nil {
- return nil, err
- }
case appsv1alpha1.Stateful, appsv1alpha1.Consensus, appsv1alpha1.Replication:
if err := workloadProcessor(
func(envConfig *corev1.ConfigMap) (client.Object, error) {
@@ -185,14 +150,6 @@ func buildComponentResources(reqCtx intctrlutil.RequestCtx, cli client.Client,
return resources, nil
}
-// TODO multi roles with same accessMode support
-func addLeaderSelectorLabels(service *corev1.Service, component *component.SynthesizedComponent) {
- leader := component.ConsensusSpec.Leader
- if len(leader.Name) > 0 {
- service.Spec.Selector[constant.RoleLabelKey] = leader.Name
- }
-}
-
var _ = Describe("Cluster Controller", func() {
cleanEnv := func() {
@@ -251,7 +208,7 @@ var _ = Describe("Cluster Controller", func() {
GetObject()
})
- It("should construct env, headless service, deployment and external service objects", func() {
+ It("should construct pdb", func() {
reqCtx := intctrlutil.RequestCtx{
Ctx: ctx,
Log: logger,
@@ -272,10 +229,6 @@ var _ = Describe("Cluster Controller", func() {
expects := []string{
"PodDisruptionBudget",
- "Service",
- "ConfigMap",
- "Service",
- "Deployment",
}
Expect(resources).Should(HaveLen(len(expects)))
for i, v := range expects {
@@ -323,9 +276,7 @@ var _ = Describe("Cluster Controller", func() {
expects := []string{
"PodDisruptionBudget",
- "Service",
"ConfigMap",
- "Service",
"StatefulSet",
}
Expect(resources).Should(HaveLen(len(expects)))
@@ -386,9 +337,7 @@ var _ = Describe("Cluster Controller", func() {
expects := []string{
"PodDisruptionBudget",
- "Service",
"ConfigMap",
- "Service",
"StatefulSet",
}
Expect(resources).Should(HaveLen(len(expects)))
@@ -448,9 +397,7 @@ var _ = Describe("Cluster Controller", func() {
expects := []string{
"PodDisruptionBudget",
- "Service",
"ConfigMap",
- "Service",
"StatefulSet",
}
Expect(resources).Should(HaveLen(len(expects)))
@@ -459,7 +406,7 @@ var _ = Describe("Cluster Controller", func() {
if isStatefulSet(v) {
sts := resources[i].(*appsv1.StatefulSet)
podSpec := sts.Spec.Template.Spec
- Expect(len(podSpec.Containers)).Should(Equal(4))
+ Expect(len(podSpec.Containers)).Should(Equal(3))
}
}
originPodSpec := clusterDef.Spec.ComponentDefs[0].PodSpec
@@ -513,9 +460,7 @@ var _ = Describe("Cluster Controller", func() {
Expect(err).Should(Succeed())
expects := []string{
"PodDisruptionBudget",
- "Service",
"ConfigMap",
- "Service",
"StatefulSet",
}
Expect(resources).Should(HaveLen(len(expects)))
@@ -551,7 +496,7 @@ var _ = Describe("Cluster Controller", func() {
GetObject()
})
- It("should construct env, headless service, statefuset object, besides an external service object", func() {
+ It("should construct env, statefuset object", func() {
reqCtx := intctrlutil.RequestCtx{
Ctx: ctx,
Log: logger,
@@ -570,14 +515,10 @@ var _ = Describe("Cluster Controller", func() {
resources, err := buildComponentResources(reqCtx, testCtx.Cli, clusterDef, clusterVersion, cluster, component)
Expect(err).Should(Succeed())
- // REVIEW: (free6om)
- // missing connection credential, TLS secret objs check?
- Expect(resources).Should(HaveLen(5))
+ Expect(resources).Should(HaveLen(3))
Expect(reflect.TypeOf(resources[0]).String()).Should(ContainSubstring("PodDisruptionBudget"))
- Expect(reflect.TypeOf(resources[1]).String()).Should(ContainSubstring("Service"))
- Expect(reflect.TypeOf(resources[2]).String()).Should(ContainSubstring("ConfigMap"))
- Expect(reflect.TypeOf(resources[3]).String()).Should(ContainSubstring("Service"))
- Expect(reflect.TypeOf(resources[4]).String()).Should(ContainSubstring("StatefulSet"))
+ Expect(reflect.TypeOf(resources[1]).String()).Should(ContainSubstring("ConfigMap"))
+ Expect(reflect.TypeOf(resources[2]).String()).Should(ContainSubstring("StatefulSet"))
})
})
diff --git a/internal/controller/plan/restore.go b/internal/controller/plan/restore.go
index b85676235f1..1cb521b6f92 100644
--- a/internal/controller/plan/restore.go
+++ b/internal/controller/plan/restore.go
@@ -21,20 +21,15 @@ package plan
import (
"context"
- "encoding/json"
- "errors"
"fmt"
- "sort"
- "strconv"
"strings"
- "time"
- batchv1 "k8s.io/api/batch/v1"
corev1 "k8s.io/api/core/v1"
apierrors "k8s.io/apimachinery/pkg/api/errors"
metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
k8sruntime "k8s.io/apimachinery/pkg/runtime"
"k8s.io/apimachinery/pkg/types"
+ "k8s.io/apimachinery/pkg/util/json"
"sigs.k8s.io/controller-runtime/pkg/client"
"sigs.k8s.io/controller-runtime/pkg/controller/controllerutil"
@@ -56,827 +51,303 @@ type RestoreManager struct {
Scheme *k8sruntime.Scheme
// private
- namespace string
- restoreTime *metav1.Time
- sourceCluster string
-}
-
-func NewRestoreManager(ctx context.Context, cli client.Client, cluster *appsv1alpha1.Cluster, scheme *k8sruntime.Scheme) *RestoreManager {
+ namespace string
+ restoreTime string
+ volumeManagementPolicy dpv1alpha1.VolumeClaimManagementPolicy
+ startingIndex int32
+ replicas int32
+ restoreLabels map[string]string
+}
+
+func NewRestoreManager(ctx context.Context,
+ cli client.Client,
+ cluster *appsv1alpha1.Cluster,
+ scheme *k8sruntime.Scheme,
+ restoreLabels map[string]string,
+ replicas, startingIndex int32,
+) *RestoreManager {
return &RestoreManager{
- Cluster: cluster,
- Client: cli,
- Ctx: ctx,
- Scheme: scheme,
+ Cluster: cluster,
+ Client: cli,
+ Ctx: ctx,
+ Scheme: scheme,
+ replicas: replicas,
+ startingIndex: startingIndex,
+ namespace: cluster.Namespace,
+ volumeManagementPolicy: dpv1alpha1.ParallelManagementPolicy,
+ restoreLabels: restoreLabels,
}
}
-const (
- backupVolumePATH = "/backupdata"
-)
-
-// DoRestore prepares restore jobs
-func DoRestore(ctx context.Context, cli client.Client, cluster *appsv1alpha1.Cluster,
- component *component.SynthesizedComponent, schema *k8sruntime.Scheme) error {
- if cluster.Status.ObservedGeneration > 1 {
- return nil
- }
-
- mgr := NewRestoreManager(ctx, cli, cluster, schema)
-
- // check restore from backup
- backupObj, err := mgr.getBackupObjectFromAnnotation(component)
+func (r *RestoreManager) DoRestore(comp *component.SynthesizedComponent) error {
+ backupObj, err := r.initFromAnnotation(comp)
if err != nil {
return err
}
if backupObj == nil {
return nil
}
-
- if err = mgr.createDataPVCs(component, backupObj); err != nil {
- return err
+ if backupObj.Status.BackupMethod == nil {
+ return intctrlutil.NewErrorf(intctrlutil.ErrorTypeRestoreFailed, `status.backupMethod of backup "%s" can not be empty`, backupObj.Name)
}
- jobs := make([]client.Object, 0)
- if backupObj.Spec.BackupType == dpv1alpha1.BackupTypeDataFile {
- restoreJobs, err := mgr.doFullBackupRestore(component, backupObj)
- if err != nil {
- return err
- }
- jobs = append(jobs, restoreJobs...)
+ if err = r.DoPrepareData(comp, backupObj); err != nil {
+ return err
}
-
- // create and waiting job finished
- if err = mgr.createJobsAndWaiting(jobs); err != nil {
+ if err = r.DoPostReady(comp, backupObj); err != nil {
return err
}
-
// do clean up
- if err = mgr.cleanupClusterAnnotations(); err != nil {
+ if err = r.cleanupClusterAnnotations(); err != nil {
return err
}
- return mgr.cleanupJobs(jobs)
+ return r.cleanupRestores(comp)
}
-// DoPITR prepares PITR jobs
-func DoPITR(ctx context.Context, cli client.Client, cluster *appsv1alpha1.Cluster,
- component *component.SynthesizedComponent, schema *k8sruntime.Scheme) error {
- if cluster.Status.ObservedGeneration > 1 {
- return nil
- }
- pitrMgr := NewRestoreManager(ctx, cli, cluster, schema)
- if need, err := pitrMgr.checkPITRAndInit(component.Name); err != nil {
- return err
- } else if !need {
- return nil
- }
-
- // get the latest base backup from point in time
- baseBackup, err := pitrMgr.getLatestBaseBackup(component.Name)
+func (r *RestoreManager) DoPrepareData(comp *component.SynthesizedComponent, backupObj *dpv1alpha1.Backup) error {
+ restore, err := r.BuildPrepareDataRestore(comp, backupObj)
if err != nil {
return err
}
-
- if err = pitrMgr.createDataPVCs(component, baseBackup); err != nil {
- return err
- }
-
- jobs := make([]client.Object, 0)
- if baseBackup.Spec.BackupType == dpv1alpha1.BackupTypeDataFile {
- restoreJobs, err := pitrMgr.doFullBackupRestore(component, baseBackup)
- if err != nil {
- return err
- }
- jobs = append(jobs, restoreJobs...)
- }
-
- continuousJobs, err := pitrMgr.doLogfileBackupRestore(component, baseBackup)
- if err != nil {
- return err
- }
- // do clean up
- if err = pitrMgr.cleanupClusterAnnotations(); err != nil {
- return err
- }
- jobs = append(jobs, continuousJobs...)
- return pitrMgr.cleanupJobs(jobs)
+ return r.createRestoreAndWait(restore)
}
-func (p *RestoreManager) doFullBackupRestore(component *component.SynthesizedComponent,
- backupObj *dpv1alpha1.Backup) ([]client.Object, error) {
- backupTool, err := p.getBackupTool(backupObj.Status.BackupToolName)
- if err != nil {
- return nil, err
+func (r *RestoreManager) BuildPrepareDataRestore(comp *component.SynthesizedComponent, backupObj *dpv1alpha1.Backup) (*dpv1alpha1.Restore, error) {
+ backupMethod := backupObj.Status.BackupMethod
+ if backupMethod == nil {
+ return nil, intctrlutil.NewErrorf(intctrlutil.ErrorTypeRestoreFailed, `status.backupMethod of backup "%s" can not be empty`, backupObj.Name)
}
- var jobs []client.Object
- if backupTool.Spec.Physical.GetPhysicalRestoreCommand() != nil {
- dataFileJobs, err := p.BuildDatafileRestoreJob(component, backupObj, backupTool)
- if err != nil {
- return nil, err
- }
- jobs = append(jobs, dataFileJobs...)
+ targetVolumes := backupMethod.TargetVolumes
+ if targetVolumes == nil {
+ return nil, nil
}
-
- if backupTool.Spec.Logical.GetLogicalRestoreCommand() != nil {
- logicalJobEnvs := p.buildCommonEnvs(backupObj)
- logicJobs, err := p.buildLogicRestoreJob(component, backupObj, &backupTool.Spec, logicalJobEnvs...)
- if err != nil {
- return nil, err
+ getClusterJSON := func() string {
+ clusterSpec := r.Cluster.DeepCopy()
+ clusterSpec.ObjectMeta = metav1.ObjectMeta{
+ Name: clusterSpec.GetName(),
+ UID: clusterSpec.GetUID(),
+ }
+ clusterSpec.Status = appsv1alpha1.ClusterStatus{}
+ b, _ := json.Marshal(*clusterSpec)
+ return string(b)
+ }
+
+ var templates []dpv1alpha1.RestoreVolumeClaim
+ pvcLabels := factory.BuildCommonLabels(r.Cluster, comp)
+ for _, v := range comp.VolumeClaimTemplates {
+ if !r.existVolumeSource(targetVolumes, v.Name) {
+ continue
+ }
+ pvc := &corev1.PersistentVolumeClaim{
+ ObjectMeta: metav1.ObjectMeta{
+ Name: fmt.Sprintf("%s-%s-%s", v.Name, r.Cluster.Name, comp.Name),
+ Labels: pvcLabels,
+ Annotations: map[string]string{
+ // satisfy the detection of transformer_halt_recovering.
+ constant.LastAppliedClusterAnnotationKey: getClusterJSON(),
+ },
+ },
}
- jobs = append(jobs, logicJobs...)
- }
- // do create datafile restore job and check completed
- if err = p.createJobsAndWaiting(jobs); err != nil {
- return nil, err
+ // build pvc labels
+ factory.BuildPersistentVolumeClaimLabels(comp, pvc, v.Name)
+ claimTemplate := dpv1alpha1.RestoreVolumeClaim{
+ ObjectMeta: pvc.ObjectMeta,
+ VolumeClaimSpec: v.Spec,
+ VolumeConfig: dpv1alpha1.VolumeConfig{
+ VolumeSource: v.Name,
+ },
+ }
+ templates = append(templates, claimTemplate)
}
- return jobs, nil
-}
-
-func (p *RestoreManager) doLogfileBackupRestore(component *component.SynthesizedComponent,
- baseBackup *dpv1alpha1.Backup) ([]client.Object, error) {
- sourceClusterUID := baseBackup.Labels[constant.DataProtectionLabelClusterUIDKey]
- logfileBackup, err := p.getLogfileBackup(component.Name, sourceClusterUID)
- if err != nil {
- return nil, err
+ if len(templates) == 0 {
+ return nil, nil
}
-
- recoveryInfo, err := p.getRecoveryInfo(baseBackup, logfileBackup)
+ schedulingSpec, err := r.buildSchedulingSpec(comp)
if err != nil {
return nil, err
}
- var continuousJobs []client.Object
- if len(recoveryInfo.Physical.GetPhysicalRestoreCommand()) != 0 {
- prepareDataJobs, err := p.buildPITRPhysicalRestoreJob(component, recoveryInfo, logfileBackup)
- if err != nil {
- return nil, err
- }
- continuousJobs = append(continuousJobs, prepareDataJobs...)
+ restore := &dpv1alpha1.Restore{
+ ObjectMeta: r.GetRestoreObjectMeta(comp, dpv1alpha1.PrepareData),
+ Spec: dpv1alpha1.RestoreSpec{
+ Backup: dpv1alpha1.BackupConfig{
+ Name: backupObj.Name,
+ Namespace: r.namespace,
+ },
+ RestoreTime: r.restoreTime,
+ PrepareDataConfig: &dpv1alpha1.PrepareDataConfig{
+ SchedulingSpec: schedulingSpec,
+ VolumeClaimManagementPolicy: r.volumeManagementPolicy,
+ RestoreVolumeClaimsTemplate: &dpv1alpha1.RestoreVolumeClaimsTemplate{
+ Replicas: r.replicas,
+ StartingIndex: r.startingIndex,
+ Templates: templates,
+ },
+ },
+ },
}
+ return restore, nil
+}
- if len(recoveryInfo.Logical.GetLogicalRestoreCommand()) != 0 {
- postReadyJobs, err := p.buildLogicRestoreJob(component, logfileBackup, recoveryInfo, recoveryInfo.Env...)
- if err != nil {
- return nil, err
- }
- continuousJobs = append(continuousJobs, postReadyJobs...)
+func (r *RestoreManager) DoPostReady(comp *component.SynthesizedComponent, backupObj *dpv1alpha1.Backup) error {
+ compStatus := r.Cluster.Status.Components[comp.Name]
+ if compStatus.Phase != appsv1alpha1.RunningClusterCompPhase {
+ return nil
}
-
- // do create PITR job and check completed
- if err = p.createJobsAndWaiting(continuousJobs); err != nil {
- return nil, err
+ jobActionLabels := factory.BuildCommonLabels(r.Cluster, comp)
+ if comp.WorkloadType == appsv1alpha1.Consensus || comp.WorkloadType == appsv1alpha1.Replication {
+ // TODO: use rsm constant
+ rsmAccessModeLabelKey := "rsm.workloads.kubeblocks.io/access-mode"
+ jobActionLabels[rsmAccessModeLabelKey] = string(appsv1alpha1.ReadWrite)
+ }
+ // TODO: get connect credential from backupPolicyTemplate
+ restore := &dpv1alpha1.Restore{
+ ObjectMeta: r.GetRestoreObjectMeta(comp, dpv1alpha1.PostReady),
+ Spec: dpv1alpha1.RestoreSpec{
+ Backup: dpv1alpha1.BackupConfig{
+ Name: backupObj.Name,
+ Namespace: r.namespace,
+ },
+ RestoreTime: r.restoreTime,
+ ReadyConfig: &dpv1alpha1.ReadyConfig{
+ ExecAction: &dpv1alpha1.ExecAction{
+ Target: dpv1alpha1.ExecActionTarget{
+ PodSelector: metav1.LabelSelector{
+ MatchLabels: factory.BuildCommonLabels(r.Cluster, comp),
+ },
+ },
+ },
+ JobAction: &dpv1alpha1.JobAction{
+ Target: dpv1alpha1.JobActionTarget{
+ PodSelector: metav1.LabelSelector{
+ MatchLabels: jobActionLabels,
+ },
+ },
+ },
+ },
+ },
}
- return continuousJobs, nil
+ return r.createRestoreAndWait(restore)
}
-func (p *RestoreManager) listCompletedBackups(componentName string) (backupItems []dpv1alpha1.Backup, err error) {
- backups := dpv1alpha1.BackupList{}
- if err := p.Client.List(p.Ctx, &backups,
- client.InNamespace(p.namespace),
- client.MatchingLabels(map[string]string{
- constant.AppInstanceLabelKey: p.sourceCluster,
- constant.KBAppComponentLabelKey: componentName,
- }),
- ); err != nil {
- return nil, err
+func (r *RestoreManager) buildSchedulingSpec(comp *component.SynthesizedComponent) (dpv1alpha1.SchedulingSpec, error) {
+ var err error
+ schedulingSpec := dpv1alpha1.SchedulingSpec{}
+ compSpec := r.Cluster.Spec.GetComponentByName(comp.Name)
+ affinity := component.BuildAffinity(r.Cluster, compSpec)
+ if schedulingSpec.Affinity, err = component.BuildPodAffinity(r.Cluster, affinity, comp); err != nil {
+ return schedulingSpec, err
}
-
- backupItems = []dpv1alpha1.Backup{}
- for _, b := range backups.Items {
- if b.Status.Phase == dpv1alpha1.BackupCompleted && b.Status.Manifests != nil && b.Status.Manifests.BackupLog != nil {
- backupItems = append(backupItems, b)
- }
+ schedulingSpec.TopologySpreadConstraints = component.BuildPodTopologySpreadConstraints(r.Cluster, affinity, comp)
+ if schedulingSpec.Tolerations, err = component.BuildTolerations(r.Cluster, compSpec); err != nil {
+ return schedulingSpec, err
}
- return backupItems, nil
+ return schedulingSpec, nil
}
-// sortBackups sorts by StopTime
-func (p *RestoreManager) sortBackups(backups []dpv1alpha1.Backup, reverse bool) []dpv1alpha1.Backup {
- sort.Slice(backups, func(i, j int) bool {
- if reverse {
- i, j = j, i
- }
- if backups[i].Status.Manifests.BackupLog.StopTime == nil && backups[j].Status.Manifests.BackupLog.StopTime != nil {
- return false
- }
- if backups[i].Status.Manifests.BackupLog.StopTime != nil && backups[j].Status.Manifests.BackupLog.StopTime == nil {
- return true
- }
- if backups[i].Status.Manifests.BackupLog.StopTime.Equal(backups[j].Status.Manifests.BackupLog.StopTime) {
- return backups[i].Name < backups[j].Name
- }
- return backups[i].Status.Manifests.BackupLog.StopTime.Before(backups[j].Status.Manifests.BackupLog.StopTime)
- })
- return backups
-}
-
-// getLatestBaseBackup gets the latest baseBackup
-func (p *RestoreManager) getLatestBaseBackup(componentName string) (*dpv1alpha1.Backup, error) {
- // 1. sorts reverse backups
- backups, err := p.listCompletedBackups(componentName)
- if err != nil {
- return nil, err
+func (r *RestoreManager) GetRestoreObjectMeta(comp *component.SynthesizedComponent, stage dpv1alpha1.RestoreStage) metav1.ObjectMeta {
+ name := fmt.Sprintf("%s-%s-%s-%s", r.Cluster.Name, comp.Name, r.Cluster.UID[:8], strings.ToLower(string(stage)))
+ if r.startingIndex != 0 {
+ name = fmt.Sprintf("%s-%d", name, r.startingIndex)
}
- backups = p.sortBackups(backups, true)
-
- // 2. gets the latest backup object
- var latestBackup *dpv1alpha1.Backup
- for _, item := range backups {
- if item.Spec.BackupType != dpv1alpha1.BackupTypeLogFile &&
- item.Status.Manifests.BackupLog.StopTime != nil && !p.restoreTime.Before(item.Status.Manifests.BackupLog.StopTime) {
- latestBackup = &item
- break
- }
+ if len(r.restoreLabels) == 0 {
+ r.restoreLabels = factory.BuildCommonLabels(r.Cluster, comp)
}
- if latestBackup == nil {
- return nil, errors.New("can not found latest base backup")
+ return metav1.ObjectMeta{
+ Name: name,
+ Namespace: r.Cluster.Namespace,
+ Labels: r.restoreLabels,
}
-
- return latestBackup, nil
}
-// checkPITRAndInit checks if cluster need to be restored
-func (p *RestoreManager) checkPITRAndInit(compName string) (bool, error) {
- // checks args if pitr supported
- cluster := p.Cluster
- restoreTimeStr, err := p.getComponentBackupInfoFromAnnotation(compName, constant.RestoreFromTimeAnnotationKey)
- if err != nil || restoreTimeStr == nil {
- return false, err
- }
- sourceCuster := cluster.Annotations[constant.RestoreFromSrcClusterAnnotationKey]
- if sourceCuster == "" {
- return false, errors.New("need specify a source cluster name to recovery")
- }
- restoreTime := &metav1.Time{}
- if err = restoreTime.UnmarshalQueryParameter(*restoreTimeStr); err != nil {
- return false, err
- }
- vctCount := 0
- for _, item := range cluster.Spec.ComponentSpecs {
- vctCount += len(item.VolumeClaimTemplates)
+// existVolumeSource checks if the backup.status.backupMethod.targetVolumes exists the target volume which should be restored.
+func (r *RestoreManager) existVolumeSource(targetVolumes *dpv1alpha1.TargetVolumeInfo, volumeName string) bool {
+ for _, v := range targetVolumes.Volumes {
+ if v == volumeName {
+ return true
+ }
}
- if vctCount == 0 {
- return false, errors.New("not support pitr without any volume claim templates")
+ for _, v := range targetVolumes.VolumeMounts {
+ if v.Name == volumeName {
+ return true
+ }
}
-
- // init args
- p.restoreTime = restoreTime
- p.sourceCluster = sourceCuster
- p.namespace = cluster.Namespace
- return true, nil
+ return false
}
-func (p *RestoreManager) getComponentBackupInfoFromAnnotation(compName, annotationKey string) (*string, error) {
- valueString := p.Cluster.Annotations[annotationKey]
+func (r *RestoreManager) initFromAnnotation(synthesizedComponent *component.SynthesizedComponent) (*dpv1alpha1.Backup, error) {
+ valueString := r.Cluster.Annotations[constant.RestoreFromBackupAnnotationKey]
if len(valueString) == 0 {
return nil, nil
}
- backupMap := map[string]string{}
+ backupMap := map[string]map[string]string{}
err := json.Unmarshal([]byte(valueString), &backupMap)
if err != nil {
return nil, err
}
- targetValue, ok := backupMap[compName]
+ backupSource, ok := backupMap[synthesizedComponent.Name]
if !ok {
return nil, nil
}
- return &targetValue, nil
-}
-
-func getVolumeMount(spec *dpv1alpha1.BackupToolSpec) string {
- dataVolumeMount := "/data"
- // TODO: hack it because the mount path is not explicitly specified in cluster definition
- for _, env := range spec.Env {
- if env.Name == constant.DPVolumeDataDIR {
- dataVolumeMount = env.Value
- break
- }
- }
- return dataVolumeMount
-}
-
-// getRecoveryInfo gets the pitr recovery info from baseBackup and logfileBackup
-func (p *RestoreManager) getRecoveryInfo(baseBackup, logfileBackup *dpv1alpha1.Backup) (*dpv1alpha1.BackupToolSpec, error) {
- // gets scripts from backup template
- backupTool := dpv1alpha1.BackupTool{}
- if err := p.Client.Get(p.Ctx, types.NamespacedName{
- Name: logfileBackup.Status.BackupToolName,
- }, &backupTool); err != nil {
- return nil, err
+ if namespace := backupSource[constant.BackupNamespaceKeyForRestore]; namespace != "" {
+ r.namespace = namespace
}
- // build recovery env
- headEnv := p.buildCommonEnvs(logfileBackup)
- // build env of recovery time
- spec := &backupTool.Spec
- timeFormat := p.getTimeFormat(spec.Env)
- headEnv = append(headEnv, corev1.EnvVar{Name: constant.DPKBRecoveryTime, Value: p.restoreTime.UTC().Format(timeFormat)})
- headEnv = append(headEnv, corev1.EnvVar{Name: constant.DPKBRecoveryTimestamp, Value: strconv.FormatInt(p.restoreTime.Unix(), 10)})
- // build env of backup startTime and user contexts
- if baseBackup.Status.Manifests != nil {
- // inject env for backup startTime
- backupLog := baseBackup.Status.Manifests.BackupLog
- startTime := baseBackup.Status.StartTimestamp
- if backupLog != nil && backupLog.StartTime != nil {
- startTime = backupLog.StartTime
- }
- if startTime != nil {
- startTimeEnv := corev1.EnvVar{Name: constant.DPBaseBackupStartTime, Value: startTime.UTC().Format(timeFormat)}
- startTimeTimestampEnv := corev1.EnvVar{Name: constant.DPBaseBackupStartTimestamp, Value: strconv.FormatInt(startTime.Unix(), 10)}
- headEnv = append(headEnv, startTimeEnv, startTimeTimestampEnv)
- }
- // inject env for user contexts
- backupUserContext := baseBackup.Status.Manifests.UserContext
- for k, v := range backupUserContext {
- headEnv = append(headEnv, corev1.EnvVar{Name: strings.ToUpper(k), Value: v})
- }
- }
- spec.Env = append(headEnv, spec.Env...)
- return spec, nil
-}
-
-func (p *RestoreManager) getLogfileBackup(componentName string, sourceClusterUID string) (*dpv1alpha1.Backup, error) {
- logfileBackupList := dpv1alpha1.BackupList{}
- if err := p.Client.List(p.Ctx, &logfileBackupList,
- client.MatchingLabels{
- constant.AppInstanceLabelKey: p.sourceCluster,
- constant.KBAppComponentLabelKey: componentName,
- constant.BackupTypeLabelKeyKey: string(dpv1alpha1.BackupTypeLogFile),
- }); err != nil {
- return nil, err
- }
- if len(logfileBackupList.Items) == 0 {
- return nil, errors.New("not found logfile backups")
- }
- backups := p.sortBackups(logfileBackupList.Items, true)
- for _, v := range backups {
- // filter backups with cluster uid for excluding same cluster name
- if v.Labels[constant.DataProtectionLabelClusterUIDKey] == sourceClusterUID {
- return &v, nil
- }
- }
- // TODO: return an error if logfile backup is not found after v0.7.0, return the first logfile for compatibility with version v0.5.0.
- return &logfileBackupList.Items[0], nil
-}
-
-func (p *RestoreManager) getLogfilePVC(logfileBackup *dpv1alpha1.Backup) (*corev1.PersistentVolumeClaim, error) {
- pvcKey := types.NamespacedName{
- Name: logfileBackup.Status.PersistentVolumeClaimName,
- Namespace: logfileBackup.Namespace,
- }
- pvc := corev1.PersistentVolumeClaim{}
- if err := p.Client.Get(p.Ctx, pvcKey, &pvc); err != nil {
- return nil, err
- }
- return &pvc, nil
-}
-
-func (p *RestoreManager) getDataPVCs(componentName string) ([]corev1.PersistentVolumeClaim, error) {
- dataPVCList := corev1.PersistentVolumeClaimList{}
- pvcLabels := map[string]string{
- constant.AppInstanceLabelKey: p.Cluster.Name,
- constant.KBAppComponentLabelKey: componentName,
- constant.VolumeTypeLabelKey: string(appsv1alpha1.VolumeTypeData),
- }
- if err := p.Client.List(p.Ctx, &dataPVCList,
- client.InNamespace(p.namespace),
- client.MatchingLabels(pvcLabels)); err != nil {
- return nil, err
- }
- return dataPVCList.Items, nil
-}
-
-// When the pvc has been bound on the determined pod,
-// this is a little different from the getDataPVCs function,
-// we need to get the node name of the pvc according to the pod,
-// and the job must be the same as the node name of the pvc
-func (p *RestoreManager) getDataPVCsAndPods(componentName string, podRestoreScope dpv1alpha1.PodRestoreScope) (map[string]corev1.Pod, error) {
- podList := corev1.PodList{}
- podLabels := map[string]string{
- constant.AppInstanceLabelKey: p.Cluster.Name,
- constant.KBAppComponentLabelKey: componentName,
- }
- if err := p.Client.List(p.Ctx, &podList,
- client.InNamespace(p.namespace),
- client.MatchingLabels(podLabels)); err != nil {
- return nil, err
- }
- dataPVCsAndPodsMap := map[string]corev1.Pod{}
- for _, targetPod := range podList.Items {
- for _, volume := range targetPod.Spec.Volumes {
- if volume.PersistentVolumeClaim == nil {
- continue
- }
- dataPVC := corev1.PersistentVolumeClaim{}
- pvcKey := types.NamespacedName{Namespace: targetPod.Namespace, Name: volume.PersistentVolumeClaim.ClaimName}
- if err := p.Client.Get(p.Ctx, pvcKey, &dataPVC); err != nil {
- return nil, err
- }
- if dataPVC.Labels[constant.VolumeTypeLabelKey] != string(appsv1alpha1.VolumeTypeData) {
- continue
- }
- if podRestoreScope == dpv1alpha1.PodRestoreScopeAll {
- dataPVCsAndPodsMap[dataPVC.Name] = targetPod
- continue
- }
- if podRestoreScope == dpv1alpha1.PodRestoreScopeReadWrite {
- if targetPod.Labels[constant.ConsensusSetAccessModeLabelKey] == string(appsv1alpha1.ReadWrite) ||
- targetPod.Labels[constant.RoleLabelKey] == constant.Primary {
- dataPVCsAndPodsMap[dataPVC.Name] = targetPod
- break
- }
- }
- }
- }
- return dataPVCsAndPodsMap, nil
-}
-
-func (p *RestoreManager) getDataVCT(synthesizedComponent *component.SynthesizedComponent) corev1.PersistentVolumeClaimTemplate {
- vctMap := map[string]corev1.PersistentVolumeClaimTemplate{}
- for _, vct := range synthesizedComponent.VolumeClaimTemplates {
- vctMap[vct.Name] = vct
- }
- for _, vt := range synthesizedComponent.VolumeTypes {
- if vt.Type == appsv1alpha1.VolumeTypeData {
- return vctMap[vt.Name]
-
- }
- }
- if len(synthesizedComponent.VolumeClaimTemplates) != 0 {
- return synthesizedComponent.VolumeClaimTemplates[0]
- }
- return corev1.PersistentVolumeClaimTemplate{}
-}
-
-func (p *RestoreManager) createDataPVCs(synthesizedComponent *component.SynthesizedComponent, backup *dpv1alpha1.Backup) error {
- // determines the data volume type
- vct := p.getDataVCT(synthesizedComponent)
- if vct.Name == "" {
- return intctrlutil.NewNotFound("can not found any PersistentVolumeClaim of data type")
- }
-
- snapshotName := ""
- if backup != nil && backup.Spec.BackupType == dpv1alpha1.BackupTypeSnapshot {
- snapshotName = backup.Name
- }
- for i := int32(0); i < synthesizedComponent.Replicas; i++ {
- pvcName := fmt.Sprintf("%s-%s-%s-%d", vct.Name, p.Cluster.Name, synthesizedComponent.Name, i)
- pvcKey := types.NamespacedName{Namespace: p.Cluster.Namespace, Name: pvcName}
- pvc, err := factory.BuildPVC(p.Cluster, synthesizedComponent, &vct, pvcKey, snapshotName)
- if err != nil {
- return err
- }
- // Prevents halt recovery from checking uncleaned resources
- if pvc.Annotations == nil {
- pvc.Annotations = map[string]string{}
- }
- pvc.Annotations[constant.LastAppliedClusterAnnotationKey] =
- fmt.Sprintf(`{"metadata":{"uid":"%s","name":"%s"}}`, p.Cluster.UID, p.Cluster.Name)
-
- if err = p.Client.Create(p.Ctx, pvc); err != nil && !apierrors.IsAlreadyExists(err) {
- return err
- }
- }
- return nil
-}
-
-func (p *RestoreManager) getBackupObjectFromAnnotation(synthesizedComponent *component.SynthesizedComponent) (*dpv1alpha1.Backup, error) {
- backupSourceName, err := p.getComponentBackupInfoFromAnnotation(synthesizedComponent.Name, constant.RestoreFromBackUpAnnotationKey)
- if backupSourceName == nil || err != nil {
- return nil, err
+ if managementPolicy := backupSource[constant.VolumeManagementPolicyKeyForRestore]; managementPolicy != "" {
+ r.volumeManagementPolicy = dpv1alpha1.VolumeClaimManagementPolicy(managementPolicy)
}
+ r.restoreTime = backupSource[constant.RestoreTimeKeyForRestore]
backup := &dpv1alpha1.Backup{}
- if err = p.Client.Get(p.Ctx, types.NamespacedName{Name: *backupSourceName, Namespace: p.Cluster.Namespace}, backup); err != nil {
+ if err = r.Client.Get(r.Ctx, types.NamespacedName{Name: backupSource[constant.BackupNameKeyForRestore], Namespace: r.Cluster.Namespace}, backup); err != nil {
return nil, err
}
return backup, nil
}
-func (p *RestoreManager) BuildDatafileRestoreJob(synthesizedComponent *component.SynthesizedComponent, backup *dpv1alpha1.Backup, backupTool *dpv1alpha1.BackupTool) (objs []client.Object, err error) {
- pvcNames := make([]string, 0)
- vct := p.getDataVCT(synthesizedComponent)
- for i := int32(0); i < synthesizedComponent.Replicas; i++ {
- pvcNames = append(pvcNames, fmt.Sprintf("%s-%s-%s-%d", vct.Name, p.Cluster.Name, synthesizedComponent.Name, i))
- }
- return p.BuildDatafileRestoreJobByPVCS(synthesizedComponent, backup, backupTool, pvcNames, p.buildCommonLabels(synthesizedComponent))
-}
-
-func (p *RestoreManager) BuildDatafileRestoreJobByPVCS(synthesizedComponent *component.SynthesizedComponent,
- backup *dpv1alpha1.Backup,
- backupTool *dpv1alpha1.BackupTool,
- pvcNames []string,
- labels map[string]string) (objs []client.Object, err error) {
-
- // build volumes and volumeMounts of backup
- buildBackupVolumesAndMounts := func() ([]corev1.Volume, []corev1.VolumeMount, string, string) {
- backupPVCName := backup.Status.PersistentVolumeClaimName
- // builds datafile volumes and volumeMounts
- backupVolumeName := fmt.Sprintf("%s-%s", synthesizedComponent.Name, backupPVCName)
- backupVolumes := []corev1.Volume{
- {
- Name: backupVolumeName,
- VolumeSource: corev1.VolumeSource{PersistentVolumeClaim: &corev1.PersistentVolumeClaimVolumeSource{ClaimName: backupPVCName}},
- },
- }
- backupMountPath := "/" + backup.Name
- backupVolumeMounts := []corev1.VolumeMount{{Name: backupVolumeName, MountPath: backupMountPath}}
-
- // build logfile volumes and volumeMounts
- logFilePVCName := backup.Status.LogFilePersistentVolumeClaimName
- if !backupTool.Spec.Physical.IsRelyOnLogfile() || logFilePVCName == backupPVCName {
- return backupVolumes, backupVolumeMounts, backupMountPath, backupMountPath
- }
- logFileVolumeName := fmt.Sprintf("%s-%s", synthesizedComponent.Name, logFilePVCName)
- backupVolumes = append(backupVolumes, corev1.Volume{
- Name: logFileVolumeName,
- VolumeSource: corev1.VolumeSource{PersistentVolumeClaim: &corev1.PersistentVolumeClaimVolumeSource{ClaimName: logFilePVCName}},
- })
- logfileMountPath := "/" + backup.Name + "-logfile"
- backupVolumeMounts = append(backupVolumeMounts, corev1.VolumeMount{Name: logFileVolumeName, MountPath: logfileMountPath})
- return backupVolumes, backupVolumeMounts, backupMountPath, logfileMountPath
- }
-
- backupVolumes, backupVolumeMounts, backupMountPath, logfileMountPath := buildBackupVolumesAndMounts()
- backupVolumeMounts = append(backupVolumeMounts, synthesizedComponent.PodSpec.Containers[0].VolumeMounts...)
- volumeMountMap := map[string]corev1.VolumeMount{}
- for _, mount := range backupVolumeMounts {
- volumeMountMap[mount.Name] = mount
- }
-
- // builds env
- buildEnv := func() []corev1.EnvVar {
- env := []corev1.EnvVar{{Name: constant.DPBackupName, Value: backup.Name}}
- manifests := backup.Status.Manifests
- if manifests != nil && manifests.BackupTool != nil {
- env = append(env, corev1.EnvVar{Name: constant.DPBackupDIR, Value: fmt.Sprintf("%s%s", backupMountPath, manifests.BackupTool.FilePath)})
- if manifests.BackupTool.LogFilePath != "" {
- env = append(env, corev1.EnvVar{Name: constant.DPLogFileDIR, Value: fmt.Sprintf("%s%s", logfileMountPath, manifests.BackupTool.LogFilePath)})
- }
- }
- timeFormat := p.getTimeFormat(backupTool.Spec.Env)
- stopTime := backup.Status.GetStopTime()
- if stopTime != nil {
- env = append(env, corev1.EnvVar{Name: constant.DPBackupStopTime, Value: stopTime.Format(timeFormat)})
- }
- // merges env from backup tool.
- env = append(env, backupTool.Spec.Env...)
- return env
- }
- env := buildEnv()
-
- objs = make([]client.Object, 0)
- vct := p.getDataVCT(synthesizedComponent)
- for _, pvcName := range pvcNames {
- dataVolume := corev1.Volume{
- Name: vct.Name,
- VolumeSource: corev1.VolumeSource{PersistentVolumeClaim: &corev1.PersistentVolumeClaimVolumeSource{ClaimName: pvcName}},
- }
- volumes := []corev1.Volume{dataVolume}
- volumes = append(volumes, backupVolumes...)
- volumes = append(volumes, synthesizedComponent.PodSpec.Volumes...)
- volumeMounts := make([]corev1.VolumeMount, 0)
- for _, volume := range volumes {
- if vmount, ok := volumeMountMap[volume.Name]; ok {
- volumeMounts = append(volumeMounts, vmount)
- }
- }
- jobName := p.GetDatafileRestoreJobName(pvcName)
- job, err := factory.BuildRestoreJob(p.Cluster, synthesizedComponent, jobName, backupTool.Spec.Image,
- backupTool.Spec.Physical.GetPhysicalRestoreCommand(), volumes, volumeMounts, env, backupTool.Spec.Resources)
- if err != nil {
- return nil, err
- }
- // if the workload uses local pv, the job's affinity should consistent with workload.
- // so datafile job should contain cluster affinity constraints.
- affinity := component.BuildAffinity(p.Cluster, p.Cluster.Spec.GetComponentByName(synthesizedComponent.Name))
- if job.Spec.Template.Spec.Affinity, err = component.BuildPodAffinity(p.Cluster, affinity, synthesizedComponent); err != nil {
- return nil, err
- }
- if p.Scheme != nil {
- if err = controllerutil.SetControllerReference(p.Cluster, job, p.Scheme); err != nil {
- return nil, err
- }
- }
- job.SetLabels(labels)
- objs = append(objs, job)
- }
- return objs, nil
-}
-
-func (p *RestoreManager) buildPITRPhysicalRestoreJob(synthesizedComponent *component.SynthesizedComponent,
- recoveryInfo *dpv1alpha1.BackupToolSpec,
- logfileBackup *dpv1alpha1.Backup) (objs []client.Object, err error) {
- // gets data dir pvc name
- dataPVCs, err := p.getDataPVCs(synthesizedComponent.Name)
- if err != nil {
- return objs, err
- }
- if len(dataPVCs) == 0 {
- return objs, errors.New("not found data pvc")
- }
- // renders the pitrJob cue template
- image := recoveryInfo.Image
- if image == "" {
- image = synthesizedComponent.PodSpec.Containers[0].Image
- }
- logfilePVC, err := p.getLogfilePVC(logfileBackup)
- if err != nil {
- return objs, err
- }
- dataVolumeMount := getVolumeMount(recoveryInfo)
- volumeMounts := []corev1.VolumeMount{
- {Name: "data", MountPath: dataVolumeMount},
- {Name: "log", MountPath: backupVolumePATH},
- }
- // creates physical restore job
- for _, dataPVC := range dataPVCs {
- volumes := []corev1.Volume{
- {Name: "data", VolumeSource: corev1.VolumeSource{
- PersistentVolumeClaim: &corev1.PersistentVolumeClaimVolumeSource{ClaimName: dataPVC.GetName()}}},
- {Name: "log", VolumeSource: corev1.VolumeSource{
- PersistentVolumeClaim: &corev1.PersistentVolumeClaimVolumeSource{ClaimName: logfilePVC.GetName()}}},
- }
- pitrJobName := p.buildRestoreJobName(fmt.Sprintf("pitr-phy-%s", dataPVC.GetName()))
- pitrJob, err := factory.BuildRestoreJob(p.Cluster, synthesizedComponent, pitrJobName, image,
- recoveryInfo.Physical.GetPhysicalRestoreCommand(), volumes, volumeMounts, recoveryInfo.Env, recoveryInfo.Resources)
- if err != nil {
- return objs, err
- }
- if p.Scheme != nil {
- if err = controllerutil.SetControllerReference(p.Cluster, pitrJob, p.Scheme); err != nil {
- return nil, err
- }
- }
- pitrJob.SetLabels(p.buildCommonLabels(synthesizedComponent))
- // collect pvcs and jobs for later deletion
- objs = append(objs, pitrJob)
- }
-
- return objs, nil
-}
-
-func (p *RestoreManager) buildLogicRestoreJob(synthesizedComponent *component.SynthesizedComponent,
- backup *dpv1alpha1.Backup,
- backupToolSpec *dpv1alpha1.BackupToolSpec,
- envs ...corev1.EnvVar) (objs []client.Object, err error) {
- // creates logic restore job, usually imported after the cluster service is started
- if p.Cluster.Status.Phase != appsv1alpha1.RunningClusterPhase {
- return nil, nil
- }
- image := backupToolSpec.Image
- if image == "" {
- image = synthesizedComponent.PodSpec.Containers[0].Image
- }
- dataVolumeMount := getVolumeMount(backupToolSpec)
- volumeMounts := []corev1.VolumeMount{
- {Name: "data", MountPath: dataVolumeMount},
- {Name: "backup-data", MountPath: backupVolumePATH},
+// createRestoreAndWait create the restore CR and wait for completion.
+func (r *RestoreManager) createRestoreAndWait(restore *dpv1alpha1.Restore) error {
+ if restore == nil {
+ return nil
}
- pvcsAndPodsMap, err := p.getDataPVCsAndPods(synthesizedComponent.Name, backupToolSpec.Logical.PodScope)
- if err != nil {
- return objs, err
- }
- jobEnv := backupToolSpec.Env
- jobEnv = append(jobEnv, envs...)
- for pvcName, pod := range pvcsAndPodsMap {
- podENV := pod.Spec.Containers[0].Env
- podENV = append(podENV, corev1.EnvVar{Name: constant.DPDBHost, Value: intctrlutil.BuildPodHostDNS(&pod)})
- podENV = append(podENV, jobEnv...)
- volumes := []corev1.Volume{
- {Name: "data", VolumeSource: corev1.VolumeSource{
- PersistentVolumeClaim: &corev1.PersistentVolumeClaimVolumeSource{ClaimName: pvcName}}},
- {Name: "backup-data", VolumeSource: corev1.VolumeSource{
- PersistentVolumeClaim: &corev1.PersistentVolumeClaimVolumeSource{ClaimName: backup.Status.PersistentVolumeClaimName}}},
- }
- logicJobName := p.buildRestoreJobName(fmt.Sprintf("restore-%s-logic-%s", backup.Spec.BackupType, pod.Name))
- logicJob, err := factory.BuildRestoreJob(p.Cluster, synthesizedComponent, logicJobName, image,
- backupToolSpec.Logical.GetLogicalRestoreCommand(), volumes, volumeMounts, podENV, backupToolSpec.Resources)
- if err != nil {
- return objs, err
- }
- if p.Scheme != nil {
- if err = controllerutil.SetControllerReference(p.Cluster, logicJob, p.Scheme); err != nil {
- return nil, err
- }
- }
- logicJob.SetLabels(p.buildCommonLabels(synthesizedComponent))
- // DO NOT use "volume.kubernetes.io/selected-node" annotation key in PVC, because it is unreliable.
- logicJob.Spec.Template.Spec.NodeName = pod.Spec.NodeName
- objs = append(objs, logicJob)
+ if r.Scheme != nil {
+ _ = controllerutil.SetControllerReference(r.Cluster, restore, r.Scheme)
}
-
- return objs, nil
-}
-
-func (p *RestoreManager) checkJobDone(key client.ObjectKey) (bool, error) {
- result := &batchv1.Job{}
- if err := p.Client.Get(p.Ctx, key, result); err != nil {
- if apierrors.IsNotFound(err) {
- return false, nil
- }
- // if err is NOT "not found", that means unknown error.
- return false, err
- }
- if result.Status.Conditions != nil && len(result.Status.Conditions) > 0 {
- jobStatusCondition := result.Status.Conditions[0]
- if jobStatusCondition.Type == batchv1.JobComplete {
- return true, nil
- } else if jobStatusCondition.Type == batchv1.JobFailed {
- return true, errors.New(jobStatusCondition.Reason)
- }
- }
- // if found, return true
- return false, nil
-}
-
-func (p *RestoreManager) createJobsAndWaiting(objs []client.Object) error {
- // creates and checks into different loops to support concurrent resource creation.
- for _, job := range objs {
- fetchedJob := &batchv1.Job{}
- if err := p.Client.Get(p.Ctx, client.ObjectKeyFromObject(job), fetchedJob); err != nil {
- if !apierrors.IsNotFound(err) {
- return err
- }
- if err = p.Client.Create(p.Ctx, job); err != nil && !apierrors.IsAlreadyExists(err) {
- return err
- }
+ if err := r.Client.Get(r.Ctx, client.ObjectKeyFromObject(restore), restore); err != nil {
+ if !apierrors.IsNotFound(err) {
+ return err
}
- }
- for _, job := range objs {
- if done, err := p.checkJobDone(client.ObjectKeyFromObject(job)); err != nil {
+ if err = r.Client.Create(r.Ctx, restore); err != nil && !apierrors.IsAlreadyExists(err) {
return err
- } else if !done {
- return intctrlutil.NewErrorf(intctrlutil.ErrorTypeNeedWaiting, "waiting restore job %s", job.GetName())
}
}
- return nil
-}
-
-func (p *RestoreManager) cleanupJobs(objs []client.Object) error {
- if p.Cluster.Status.Phase == appsv1alpha1.RunningClusterPhase {
- for _, obj := range objs {
- if err := intctrlutil.BackgroundDeleteObject(p.Client, p.Ctx, obj); err != nil {
- return err
- }
- }
+ if restore.Status.Phase == dpv1alpha1.RestorePhaseCompleted {
+ return nil
}
- return nil
+ if restore.Status.Phase == dpv1alpha1.RestorePhaseFailed {
+ return intctrlutil.NewErrorf(intctrlutil.ErrorTypeRestoreFailed, `restore "%s" is Failed, you can describe it and re-restore the cluster.`, restore.GetName())
+ }
+ return intctrlutil.NewErrorf(intctrlutil.ErrorTypeNeedWaiting, `waiting for restore "%s" successfully`, restore.GetName())
}
-func (p *RestoreManager) cleanupClusterAnnotations() error {
- if p.Cluster.Status.Phase == appsv1alpha1.RunningClusterPhase && p.Cluster.Annotations != nil {
- cluster := p.Cluster
+func (r *RestoreManager) cleanupClusterAnnotations() error {
+ if r.Cluster.Status.Phase == appsv1alpha1.RunningClusterPhase && r.Cluster.Annotations != nil {
+ cluster := r.Cluster
patch := client.MergeFrom(cluster.DeepCopy())
- delete(cluster.Annotations, constant.RestoreFromSrcClusterAnnotationKey)
- delete(cluster.Annotations, constant.RestoreFromTimeAnnotationKey)
- delete(cluster.Annotations, constant.RestoreFromBackUpAnnotationKey)
- return p.Client.Patch(p.Ctx, cluster, patch)
+ delete(cluster.Annotations, constant.RestoreFromBackupAnnotationKey)
+ return r.Client.Patch(r.Ctx, cluster, patch)
}
return nil
}
-// buildRestoreJobName builds the restore job name.
-func (p *RestoreManager) buildRestoreJobName(jobName string) string {
- l := len(jobName)
- if l > 63 {
- return fmt.Sprintf("%s-%s", jobName[:57], jobName[l-5:l])
- }
- return jobName
-}
-
-func (p *RestoreManager) GetDatafileRestoreJobName(pvcName string) string {
- return p.buildRestoreJobName(fmt.Sprintf("base-%s", pvcName))
-}
-
-func (p *RestoreManager) getBackupTool(backupToolName string) (*dpv1alpha1.BackupTool, error) {
- backupToolKey := client.ObjectKey{Name: backupToolName}
- backupTool := &dpv1alpha1.BackupTool{}
- if err := p.Client.Get(p.Ctx, backupToolKey, backupTool); err != nil {
- return nil, err
- }
- return backupTool, nil
-}
-
-func (p *RestoreManager) buildCommonLabels(synthesizedComponent *component.SynthesizedComponent) map[string]string {
- return map[string]string{
- constant.AppManagedByLabelKey: constant.AppName,
- constant.AppInstanceLabelKey: p.Cluster.Name,
- constant.KBAppComponentLabelKey: synthesizedComponent.Name,
- }
-}
-
-func (p *RestoreManager) buildCommonEnvs(backup *dpv1alpha1.Backup) []corev1.EnvVar {
- backupDIR := backup.Name
- if backup.Status.Manifests != nil && backup.Status.Manifests.BackupTool != nil {
- backupDIR = backup.Status.Manifests.BackupTool.FilePath
+func (r *RestoreManager) cleanupRestores(comp *component.SynthesizedComponent) error {
+ if r.Cluster.Status.Phase != appsv1alpha1.RunningClusterPhase {
+ return nil
}
- return []corev1.EnvVar{
- {Name: constant.DPBackupDIR, Value: backupVolumePATH + backupDIR},
- {Name: constant.DPBackupName, Value: backup.Name},
+ restoreList := &dpv1alpha1.RestoreList{}
+ if err := r.Client.List(r.Ctx, restoreList, client.MatchingLabels(factory.BuildCommonLabels(r.Cluster, comp))); err != nil {
+ return err
}
-}
-
-func (p *RestoreManager) getTimeFormat(envs []corev1.EnvVar) string {
- for _, env := range envs {
- if env.Name == constant.DPTimeFormat {
- return env.Value
+ for i := range restoreList.Items {
+ if err := intctrlutil.BackgroundDeleteObject(r.Client, r.Ctx, &restoreList.Items[i]); err != nil {
+ return err
}
}
- return time.RFC3339
+ return nil
}
diff --git a/internal/controller/plan/restore_test.go b/internal/controller/plan/restore_test.go
index f02452d8b8d..30d30ac22d9 100644
--- a/internal/controller/plan/restore_test.go
+++ b/internal/controller/plan/restore_test.go
@@ -25,7 +25,6 @@ import (
. "github.com/onsi/ginkgo/v2"
. "github.com/onsi/gomega"
- batchv1 "k8s.io/api/batch/v1"
"k8s.io/client-go/kubernetes/scheme"
corev1 "k8s.io/api/core/v1"
@@ -40,6 +39,7 @@ import (
intctrlutil "github.com/apecloud/kubeblocks/internal/controllerutil"
"github.com/apecloud/kubeblocks/internal/generics"
testapps "github.com/apecloud/kubeblocks/internal/testutil/apps"
+ testdp "github.com/apecloud/kubeblocks/internal/testutil/dataprotection"
)
var _ = Describe("PITR Functions", func() {
@@ -49,11 +49,10 @@ var _ = Describe("PITR Functions", func() {
var (
randomStr = testCtx.GetRandomStr()
- clusterName = "cluster-for-pitr-" + randomStr
+ clusterName = "cluster-" + randomStr
now = metav1.Now()
startTime = metav1.Time{Time: now.Add(-time.Hour * 2)}
- stopTime = metav1.Time{Time: now.Add(time.Hour * 2)}
)
cleanEnv := func() {
@@ -79,8 +78,7 @@ var _ = Describe("PITR Functions", func() {
testapps.ClearResources(&testCtx, generics.PodSignature, inNS, ml, &opts)
testapps.ClearResources(&testCtx, generics.BackupSignature, inNS, ml)
testapps.ClearResources(&testCtx, generics.BackupPolicySignature, inNS, ml)
- testapps.ClearResources(&testCtx, generics.JobSignature, inNS, ml)
- testapps.ClearResources(&testCtx, generics.CronJobSignature, inNS, ml)
+ testapps.ClearResources(&testCtx, generics.RestoreSignature, inNS, ml)
testapps.ClearResourcesWithRemoveFinalizerOption(&testCtx, generics.PersistentVolumeClaimSignature, true, inNS, ml)
//
// non-namespaced
@@ -104,21 +102,19 @@ var _ = Describe("PITR Functions", func() {
)
var (
- clusterDef *appsv1alpha1.ClusterDefinition
- clusterVersion *appsv1alpha1.ClusterVersion
- cluster *appsv1alpha1.Cluster
- synthesizedComponent *component.SynthesizedComponent
- pvc *corev1.PersistentVolumeClaim
- backup *dpv1alpha1.Backup
- fullBackupTool *dpv1alpha1.BackupTool
- fullBackupToolName string
- continuousBackupTool *dpv1alpha1.BackupTool
- continuousBackupToolName string
+ clusterDef *appsv1alpha1.ClusterDefinition
+ clusterVersion *appsv1alpha1.ClusterVersion
+ cluster *appsv1alpha1.Cluster
+ synthesizedComponent *component.SynthesizedComponent
+ pvc *corev1.PersistentVolumeClaim
+ backup *dpv1alpha1.Backup
+ fullBackupActionSet *dpv1alpha1.ActionSet
+ fullBackupActionSetName string
)
BeforeEach(func() {
clusterDef = testapps.NewClusterDefFactory(clusterDefName).
- AddComponentDef(testapps.StatefulMySQLComponent, mysqlCompType).
+ AddComponentDef(testapps.ConsensusMySQLComponent, mysqlCompType).
AddComponentDef(testapps.StatelessNginxComponent, nginxCompType).
Create(&testCtx).GetObject()
clusterVersion = testapps.NewClusterVersionFactory(clusterVersionName, clusterDefName).
@@ -132,6 +128,7 @@ var _ = Describe("PITR Functions", func() {
cluster = testapps.NewClusterFactory(testCtx.DefaultNamespace, clusterName,
clusterDef.Name, clusterVersion.Name).
AddComponent(mysqlCompName, mysqlCompType).
+ SetReplicas(3).
SetClusterAffinity(&appsv1alpha1.Affinity{
PodAntiAffinity: appsv1alpha1.Required,
TopologyKeys: []string{topologyKey},
@@ -140,7 +137,6 @@ var _ = Describe("PITR Functions", func() {
},
}).
AddVolumeClaimTemplate(testapps.DataVolumeName, pvcSpec).
- AddRestorePointInTime(metav1.Time{Time: stopTime.Time}, mysqlCompName, sourceCluster).
Create(&testCtx).GetObject()
By("By mocking a pvc")
@@ -155,33 +151,16 @@ var _ = Describe("PITR Functions", func() {
_ = testapps.NewPodFactory(testCtx.DefaultNamespace, clusterName+"-"+mysqlCompName+"-0").
AddAppInstanceLabel(clusterName).
AddAppComponentLabel(mysqlCompName).
- AddAppManangedByLabel().
+ AddAppManagedByLabel().
AddVolume(volume).
AddLabels(constant.ConsensusSetAccessModeLabelKey, string(appsv1alpha1.ReadWrite)).
AddContainer(corev1.Container{Name: testapps.DefaultMySQLContainerName, Image: testapps.ApeCloudMySQLImage}).
AddNodeName("fake-node-name").
Create(&testCtx).GetObject()
- By("create datafile backup tool")
- fullBackupTool = testapps.CreateCustomizedObj(&testCtx, "backup/backuptool.yaml", &dpv1alpha1.BackupTool{}, testapps.RandomizedObjName())
- fullBackupToolName = fullBackupTool.Name
-
- By("By creating backup tool: ")
- backupSelfDefineObj := &dpv1alpha1.BackupTool{}
- backupSelfDefineObj.SetLabels(map[string]string{
- constant.BackupToolTypeLabelKey: "pitr",
- constant.ClusterDefLabelKey: clusterDefName,
- })
- continuousBackupTool = testapps.CreateCustomizedObj(&testCtx, "backup/pitr_backuptool.yaml",
- backupSelfDefineObj, testapps.RandomizedObjName())
- // set datafile backup relies on logfile
- Expect(testapps.ChangeObj(&testCtx, continuousBackupTool, func(tmpObj *dpv1alpha1.BackupTool) {
- tmpObj.Spec.Physical.RelyOnLogfile = true
- })).Should(Succeed())
- continuousBackupToolName = continuousBackupTool.Name
-
- backupObj := dpv1alpha1.BackupToolList{}
- Expect(testCtx.Cli.List(testCtx.Ctx, &backupObj)).Should(Succeed())
+ By("create actionset of full backup")
+ fullBackupActionSet = testapps.CreateCustomizedObj(&testCtx, "backup/actionset.yaml", &dpv1alpha1.ActionSet{}, testapps.RandomizedObjName())
+ fullBackupActionSetName = fullBackupActionSet.Name
By("By creating backup policyTemplate: ")
backupTplLabels := map[string]string{
@@ -191,17 +170,13 @@ var _ = Describe("PITR Functions", func() {
WithRandomName().SetLabels(backupTplLabels).
AddBackupPolicy(mysqlCompName).
SetClusterDefRef(clusterDefName).
- SetTTL(defaultTTL).
- AddDatafilePolicy().
- SetBackupToolName(fullBackupToolName).
- SetSchedule("0 * * * *", true).
- AddIncrementalPolicy().
- SetBackupToolName(continuousBackupToolName).
- SetSchedule("0 * * * *", true).
- Create(&testCtx).GetObject()
+ SetRetentionPeriod(defaultTTL).
+ AddBackupMethod(testdp.BackupMethodName, false, fullBackupActionSetName).
+ SetBackupMethodVolumeMounts(testapps.DataVolumeName, "/data")
clusterCompDefObj := clusterDef.Spec.ComponentDefs[0]
synthesizedComponent = &component.SynthesizedComponent{
+ WorkloadType: appsv1alpha1.Consensus,
PodSpec: clusterCompDefObj.PodSpec,
Probes: clusterCompDefObj.Probes,
LogConfigs: clusterCompDefObj.LogConfigs,
@@ -217,216 +192,79 @@ var _ = Describe("PITR Functions", func() {
SetStorage("1Gi").
Create(&testCtx).GetObject()
- logfileRemotePVC := testapps.NewPersistentVolumeClaimFactory(
- testCtx.DefaultNamespace, "remote-pvc-logfile", clusterName, mysqlCompName, "log").
- SetStorage("1Gi").
- Create(&testCtx).GetObject()
-
By("By creating base backup: ")
backupLabels := map[string]string{
- constant.AppInstanceLabelKey: sourceCluster,
- constant.KBAppComponentLabelKey: mysqlCompName,
- constant.BackupTypeLabelKeyKey: string(dpv1alpha1.BackupTypeDataFile),
- constant.DataProtectionLabelClusterUIDKey: string(cluster.UID),
+ constant.AppInstanceLabelKey: sourceCluster,
+ constant.KBAppComponentLabelKey: mysqlCompName,
}
- backup = testapps.NewBackupFactory(testCtx.DefaultNamespace, backupName).
+ backup = testdp.NewBackupFactory(testCtx.DefaultNamespace, backupName).
WithRandomName().SetLabels(backupLabels).
SetBackupPolicyName("test-fake").
- SetBackupType(dpv1alpha1.BackupTypeDataFile).
+ SetBackupMethod(testdp.VSBackupMethodName).
Create(&testCtx).GetObject()
baseStartTime := &startTime
baseStopTime := &now
- backupStatus := dpv1alpha1.BackupStatus{
- Phase: dpv1alpha1.BackupCompleted,
- StartTimestamp: baseStartTime,
- CompletionTimestamp: baseStopTime,
- BackupToolName: fullBackupToolName,
- SourceCluster: clusterName,
- PersistentVolumeClaimName: remotePVC.Name,
- LogFilePersistentVolumeClaimName: logfileRemotePVC.Name,
- Manifests: &dpv1alpha1.ManifestsStatus{
- BackupTool: &dpv1alpha1.BackupToolManifestsStatus{
- FilePath: fmt.Sprintf("/%s/%s", backup.Namespace, backup.Name),
- LogFilePath: fmt.Sprintf("/%s/%s", backup.Namespace, backup.Name+"-logfile"),
- },
- BackupLog: &dpv1alpha1.BackupLogStatus{
- StartTime: baseStartTime,
- StopTime: baseStopTime,
- },
- },
- }
- patchBackupStatus(backupStatus, client.ObjectKeyFromObject(backup))
-
- By("By creating continuous backup: ")
- logfileBackupLabels := map[string]string{
- constant.AppInstanceLabelKey: sourceCluster,
- constant.KBAppComponentLabelKey: mysqlCompName,
- constant.BackupTypeLabelKeyKey: string(dpv1alpha1.BackupTypeLogFile),
- constant.DataProtectionLabelClusterUIDKey: string(cluster.UID),
- }
- incrStartTime := &startTime
- incrStopTime := &stopTime
- logfileBackup := testapps.NewBackupFactory(testCtx.DefaultNamespace, backupName).
- WithRandomName().SetLabels(logfileBackupLabels).
- SetBackupPolicyName("test-fake").
- SetBackupType(dpv1alpha1.BackupTypeLogFile).
- Create(&testCtx).GetObject()
- backupStatus = dpv1alpha1.BackupStatus{
- Phase: dpv1alpha1.BackupCompleted,
- StartTimestamp: incrStartTime,
- CompletionTimestamp: incrStopTime,
- SourceCluster: clusterName,
- PersistentVolumeClaimName: logfileRemotePVC.Name,
- BackupToolName: continuousBackupToolName,
- Manifests: &dpv1alpha1.ManifestsStatus{
- BackupLog: &dpv1alpha1.BackupLogStatus{
- StartTime: incrStartTime,
- StopTime: incrStopTime,
- },
- },
+ backup.Status = dpv1alpha1.BackupStatus{
+ Phase: dpv1alpha1.BackupPhaseCompleted,
+ StartTimestamp: baseStartTime,
+ CompletionTimestamp: baseStopTime,
+ PersistentVolumeClaimName: remotePVC.Name,
}
- patchBackupStatus(backupStatus, client.ObjectKeyFromObject(logfileBackup))
+ testdp.MockBackupStatusMethod(backup, testapps.DataVolumeName)
+ patchBackupStatus(backup.Status, client.ObjectKeyFromObject(backup))
})
It("Test restore", func() {
- By("restore from snapshot backup")
- backupSnapshot := testapps.NewBackupFactory(testCtx.DefaultNamespace, backupName).
- WithRandomName().
- SetBackupPolicyName("test-fake").
- SetBackupType(dpv1alpha1.BackupTypeSnapshot).
- Create(&testCtx).GetObject()
- restoreFromBackup := fmt.Sprintf(`{"%s":"%s"}`, mysqlCompName, backupSnapshot.Name)
- cluster.Annotations[constant.RestoreFromBackUpAnnotationKey] = restoreFromBackup
- Expect(DoRestore(ctx, testCtx.Cli, cluster, synthesizedComponent, scheme.Scheme)).Should(Succeed())
-
- By("restore from datafile backup")
- restoreFromBackup = fmt.Sprintf(`{"%s":"%s"}`, mysqlCompName, backup.Name)
- cluster.Annotations[constant.RestoreFromBackUpAnnotationKey] = restoreFromBackup
- err := DoRestore(ctx, testCtx.Cli, cluster, synthesizedComponent, scheme.Scheme)
+ By("restore from backup")
+ restoreFromBackup := fmt.Sprintf(`{"%s": {"name":"%s"}}`, mysqlCompName, backup.Name)
+ Expect(testapps.ChangeObj(&testCtx, cluster, func(tmpCluster *appsv1alpha1.Cluster) {
+ tmpCluster.Annotations = map[string]string{
+ constant.RestoreFromBackupAnnotationKey: restoreFromBackup,
+ }
+ })).Should(Succeed())
+ Expect(k8sClient.Get(ctx, client.ObjectKeyFromObject(cluster), cluster)).Should(Succeed())
+ restoreMGR := NewRestoreManager(ctx, k8sClient, cluster, scheme.Scheme, nil, 3, 0)
+ err := restoreMGR.DoRestore(synthesizedComponent)
Expect(intctrlutil.IsTargetError(err, intctrlutil.ErrorTypeNeedWaiting)).Should(BeTrue())
- })
- testPITR := func() {
- baseBackupPhysicalRestore := func() types.NamespacedName {
- By("create fullBackup physical restore job")
- err := DoPITR(ctx, testCtx.Cli, cluster, synthesizedComponent, scheme.Scheme)
- Expect(intctrlutil.IsTargetError(err, intctrlutil.ErrorTypeNeedWaiting)).Should(BeTrue())
-
- By("when base backup restore job completed")
- baseBackupJobName := fmt.Sprintf("base-%s", fmt.Sprintf("%s-%s-%s-%d", "data", clusterName, synthesizedComponent.Name, 0))
- baseBackupJobKey := types.NamespacedName{Namespace: cluster.Namespace, Name: baseBackupJobName}
- Eventually(testapps.CheckObj(&testCtx, baseBackupJobKey, func(g Gomega, fetched *batchv1.Job) {
- envs := fetched.Spec.Template.Spec.Containers[0].Env
- var existsTargetENV bool
- for _, env := range envs {
- if env.Name == constant.KBEnvPodName {
- existsTargetENV = true
- break
- }
- }
- g.Expect(existsTargetENV).Should(BeTrue())
- })).Should(Succeed())
- Eventually(testapps.GetAndChangeObjStatus(&testCtx, baseBackupJobKey, func(fetched *batchv1.Job) {
- fetched.Status.Conditions = []batchv1.JobCondition{{Type: batchv1.JobComplete}}
- })).Should(Succeed())
- return baseBackupJobKey
- }
-
- baseBackupLogicalRestore := func() types.NamespacedName {
- By("create and wait for fullbackup logical job is completed ")
- err := DoPITR(ctx, testCtx.Cli, cluster, synthesizedComponent, scheme.Scheme)
- Expect(intctrlutil.IsTargetError(err, intctrlutil.ErrorTypeNeedWaiting)).Should(BeTrue())
-
- By("when logic full backup jobs are completed")
- logicJobName := fmt.Sprintf("restore-datafile-logic-%s-%s-0", clusterName, mysqlCompName)
- logicJobKey := types.NamespacedName{Namespace: cluster.Namespace, Name: logicJobName}
- Eventually(testapps.GetAndChangeObjStatus(&testCtx, logicJobKey, func(fetched *batchv1.Job) {
- fetched.Status.Conditions = []batchv1.JobCondition{{Type: batchv1.JobComplete}}
- })).Should(Succeed())
- return logicJobKey
- }
-
- continuousPhysicalRestore := func() types.NamespacedName {
- By("create and wait for pitr physical restore job is completed ")
- err := DoPITR(ctx, testCtx.Cli, cluster, synthesizedComponent, scheme.Scheme)
- Expect(intctrlutil.IsTargetError(err, intctrlutil.ErrorTypeNeedWaiting)).Should(BeTrue())
-
- By("when physical PITR jobs are completed")
- jobName := fmt.Sprintf("pitr-phy-data-%s-%s-0", clusterName, mysqlCompName)
- jobKey := types.NamespacedName{Namespace: cluster.Namespace, Name: jobName}
- Eventually(testapps.GetAndChangeObjStatus(&testCtx, jobKey, func(fetched *batchv1.Job) {
- fetched.Status.Conditions = []batchv1.JobCondition{{Type: batchv1.JobComplete}}
- })).Should(Succeed())
- return jobKey
- }
-
- continuousLogicalRestore := func() types.NamespacedName {
- By("create and wait for pitr logical job is completed ")
- err := DoPITR(ctx, testCtx.Cli, cluster, synthesizedComponent, scheme.Scheme)
- Expect(intctrlutil.IsTargetError(err, intctrlutil.ErrorTypeNeedWaiting)).Should(BeTrue())
-
- By("mock the podScope is ReadWrite for logic restore")
- Expect(testapps.ChangeObj(&testCtx, continuousBackupTool, func(tool *dpv1alpha1.BackupTool) {
- tool.Spec.Logical.PodScope = dpv1alpha1.PodRestoreScopeReadWrite
- })).Should(Succeed())
- err = DoPITR(ctx, testCtx.Cli, cluster, synthesizedComponent, scheme.Scheme)
- Expect(intctrlutil.IsTargetError(err, intctrlutil.ErrorTypeNeedWaiting)).Should(BeTrue())
-
- By("when logic PITR jobs are completed")
- logicJobName := fmt.Sprintf("restore-logfile-logic-%s-%s-0", clusterName, mysqlCompName)
- logicJobKey := types.NamespacedName{Namespace: cluster.Namespace, Name: logicJobName}
- Eventually(testapps.GetAndChangeObjStatus(&testCtx, logicJobKey, func(fetched *batchv1.Job) {
- fetched.Status.Conditions = []batchv1.JobCondition{{Type: batchv1.JobComplete}}
- })).Should(Succeed())
- return logicJobKey
- }
- cluster.Status.ObservedGeneration = 1
- var backupJobKeys []types.NamespacedName
- // do full backup physical restore
- if fullBackupTool.Spec.Physical.GetPhysicalRestoreCommand() != nil {
- backupJobKeys = append(backupJobKeys, baseBackupPhysicalRestore())
- }
-
- // do continuous backup physical restore
- if continuousBackupTool.Spec.Physical.GetPhysicalRestoreCommand() != nil {
- backupJobKeys = append(backupJobKeys, continuousPhysicalRestore())
- }
- Expect(DoPITR(ctx, testCtx.Cli, cluster, synthesizedComponent, scheme.Scheme)).Should(Succeed())
-
- By("when logic PITR jobs are creating after cluster RUNNING")
- Eventually(testapps.GetAndChangeObjStatus(&testCtx, client.ObjectKeyFromObject(cluster), func(fetched *appsv1alpha1.Cluster) {
- fetched.Status.Phase = appsv1alpha1.RunningClusterPhase
+ By("mock restore of prepareData stage to Completed")
+ restoreMeta := restoreMGR.GetRestoreObjectMeta(synthesizedComponent, dpv1alpha1.PrepareData)
+ namedspace := types.NamespacedName{Name: restoreMeta.Name, Namespace: restoreMeta.Namespace}
+ Expect(testapps.GetAndChangeObjStatus(&testCtx, namedspace, func(restore *dpv1alpha1.Restore) {
+ restore.Status.Phase = dpv1alpha1.RestorePhaseCompleted
+ })()).ShouldNot(HaveOccurred())
+
+ By("mock cluster phase to Running")
+ Expect(testapps.ChangeObjStatus(&testCtx, cluster, func() {
+ cluster.Status.Phase = appsv1alpha1.RunningClusterPhase
+ cluster.Status.Components = map[string]appsv1alpha1.ClusterComponentStatus{
+ mysqlCompName: {
+ Phase: appsv1alpha1.RunningClusterCompPhase,
+ },
+ }
})).Should(Succeed())
- cluster.Status.Phase = appsv1alpha1.RunningClusterPhase
- // do full backup logical restore
- if fullBackupTool.Spec.Logical.GetLogicalRestoreCommand() != nil {
- backupJobKeys = append(backupJobKeys, baseBackupLogicalRestore())
- }
-
- // do continuous logical restore
- if continuousBackupTool.Spec.Logical.GetLogicalRestoreCommand() != nil {
- backupJobKeys = append(backupJobKeys, continuousLogicalRestore())
- }
- Expect(DoPITR(ctx, testCtx.Cli, cluster, synthesizedComponent, scheme.Scheme)).Should(Succeed())
-
- By("expect all jobs are cleaned")
- for _, v := range backupJobKeys {
- Eventually(testapps.CheckObjExists(&testCtx, v, &batchv1.Job{}, false)).Should(Succeed())
- }
- }
-
- It("Test PITR restore when only support physical restore for full backup", func() {
- testPITR()
- })
-
- It("Test PITR restore when only support physical logical for full backup", func() {
- Expect(testapps.ChangeObj(&testCtx, fullBackupTool, func(tool *dpv1alpha1.BackupTool) {
- fullBackupTool.Spec.Logical.RestoreCommands = fullBackupTool.Spec.Physical.GetPhysicalRestoreCommand()
- fullBackupTool.Spec.Physical.RestoreCommands = nil
+ By("wait for postReady restore created and mock it to Completed")
+ restoreMGR.Cluster = cluster
+ _ = restoreMGR.DoRestore(synthesizedComponent)
+
+ // check if restore CR of postReady stage is created.
+ restoreMeta = restoreMGR.GetRestoreObjectMeta(synthesizedComponent, dpv1alpha1.PostReady)
+ namedspace = types.NamespacedName{Name: restoreMeta.Name, Namespace: restoreMeta.Namespace}
+ Eventually(testapps.CheckObjExists(&testCtx, namedspace,
+ &dpv1alpha1.Restore{}, true)).Should(Succeed())
+ // set restore to Completed
+ Expect(testapps.GetAndChangeObjStatus(&testCtx, namedspace, func(restore *dpv1alpha1.Restore) {
+ restore.Status.Phase = dpv1alpha1.RestorePhaseCompleted
+ })()).ShouldNot(HaveOccurred())
+
+ By("clean up annotations after cluster running")
+ _ = restoreMGR.DoRestore(synthesizedComponent)
+ Eventually(testapps.CheckObj(&testCtx, client.ObjectKeyFromObject(cluster), func(g Gomega, tmpCluster *appsv1alpha1.Cluster) {
+ g.Expect(tmpCluster.Annotations[constant.RestoreFromBackupAnnotationKey]).Should(BeEmpty())
})).Should(Succeed())
- testPITR()
})
+
})
})
diff --git a/internal/controller/plan/suite_test.go b/internal/controller/plan/suite_test.go
index 20f3d6d4ba7..68e486ca709 100644
--- a/internal/controller/plan/suite_test.go
+++ b/internal/controller/plan/suite_test.go
@@ -39,7 +39,7 @@ import (
"sigs.k8s.io/controller-runtime/pkg/log/zap"
appsv1alpha1 "github.com/apecloud/kubeblocks/apis/apps/v1alpha1"
- dataprotectionv1alpha1 "github.com/apecloud/kubeblocks/apis/dataprotection/v1alpha1"
+ dpv1alpha1 "github.com/apecloud/kubeblocks/apis/dataprotection/v1alpha1"
intctrlutil "github.com/apecloud/kubeblocks/internal/controllerutil"
"github.com/apecloud/kubeblocks/internal/testutil"
viper "github.com/apecloud/kubeblocks/internal/viperx"
@@ -100,7 +100,7 @@ var _ = BeforeSuite(func() {
err = appsv1alpha1.AddToScheme(scheme.Scheme)
Expect(err).NotTo(HaveOccurred())
- err = dataprotectionv1alpha1.AddToScheme(scheme.Scheme)
+ err = dpv1alpha1.AddToScheme(scheme.Scheme)
Expect(err).NotTo(HaveOccurred())
err = snapshotv1.AddToScheme(scheme.Scheme)
diff --git a/internal/controller/rsm/enqueue_ancestor.go b/internal/controller/rsm/enqueue_ancestor.go
deleted file mode 100644
index 0cdca0127cc..00000000000
--- a/internal/controller/rsm/enqueue_ancestor.go
+++ /dev/null
@@ -1,333 +0,0 @@
-/*
-Copyright (C) 2022-2023 ApeCloud Co., Ltd
-
-This file is part of KubeBlocks project
-
-This program is free software: you can redistribute it and/or modify
-it under the terms of the GNU Affero General Public License as published by
-the Free Software Foundation, either version 3 of the License, or
-(at your option) any later version.
-
-This program is distributed in the hope that it will be useful
-but WITHOUT ANY WARRANTY; without even the implied warranty of
-MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
-GNU Affero General Public License for more details.
-
-You should have received a copy of the GNU Affero General Public License
-along with this program. If not, see .
-*/
-
-package rsm
-
-import (
- "context"
- "errors"
- "fmt"
-
- corev1 "k8s.io/api/core/v1"
- "k8s.io/apimachinery/pkg/api/meta"
- metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
- "k8s.io/apimachinery/pkg/runtime"
- "k8s.io/apimachinery/pkg/runtime/schema"
- "k8s.io/apimachinery/pkg/types"
- "k8s.io/client-go/util/workqueue"
- "sigs.k8s.io/controller-runtime/pkg/client"
- "sigs.k8s.io/controller-runtime/pkg/event"
- "sigs.k8s.io/controller-runtime/pkg/handler"
- logf "sigs.k8s.io/controller-runtime/pkg/log"
- "sigs.k8s.io/controller-runtime/pkg/reconcile"
- "sigs.k8s.io/controller-runtime/pkg/runtime/inject"
-
- roclient "github.com/apecloud/kubeblocks/internal/controller/client"
- "github.com/apecloud/kubeblocks/internal/controller/model"
-)
-
-var _ handler.EventHandler = &EnqueueRequestForAncestor{}
-
-var log = logf.FromContext(context.Background()).WithName("eventhandler").WithName("EnqueueRequestForAncestor")
-
-// EnqueueRequestForAncestor enqueues Requests for the ancestor object.
-// E.g. the ancestor object creates the StatefulSet/Deployment which then creates the Pod.
-//
-// If a ReplicatedStateMachine creates Pods, users may reconcile the ReplicatedStateMachine in response to Pod Events using:
-//
-// - a source.Kind Source with Type of Pod.
-//
-// - a EnqueueRequestForAncestor EventHandler with an OwnerType of ReplicatedStateMachine and UpToLevel set to 2.
-//
-// If source kind is corev1.Event, Event.InvolvedObject will be used as the source kind
-type EnqueueRequestForAncestor struct {
- // Client used to get owner object of
- Client roclient.ReadonlyClient
-
- // OwnerType is the type of the Owner object to look for in OwnerReferences. Only Group and Kind are compared.
- OwnerType runtime.Object
-
- // find event source up to UpToLevel
- UpToLevel int
-
- // InTypes specified the range to look for the ancestor, means all ancestors' type in the looking up tree should be in InTypes.
- // OwnerType will be included.
- // nil means only look for in OwnerType.
- InTypes []runtime.Object
-
- // groupKind is the cached Group and Kind from OwnerType
- groupKind *schema.GroupKind
-
- // ancestorGroupKinds is the cached Group and Kind from InTypes
- ancestorGroupKinds []schema.GroupKind
-
- // mapper maps GroupVersionKinds to Resources
- mapper meta.RESTMapper
-}
-
-type empty struct{}
-
-// Create implements EventHandler.
-func (e *EnqueueRequestForAncestor) Create(evt event.CreateEvent, q workqueue.RateLimitingInterface) {
- reqs := map[reconcile.Request]empty{}
- e.getOwnerReconcileRequest(evt.Object, reqs)
- for req := range reqs {
- q.Add(req)
- }
-}
-
-// Update implements EventHandler.
-func (e *EnqueueRequestForAncestor) Update(evt event.UpdateEvent, q workqueue.RateLimitingInterface) {
- reqs := map[reconcile.Request]empty{}
- e.getOwnerReconcileRequest(evt.ObjectOld, reqs)
- e.getOwnerReconcileRequest(evt.ObjectNew, reqs)
- for req := range reqs {
- q.Add(req)
- }
-}
-
-// Delete implements EventHandler.
-func (e *EnqueueRequestForAncestor) Delete(evt event.DeleteEvent, q workqueue.RateLimitingInterface) {
- reqs := map[reconcile.Request]empty{}
- e.getOwnerReconcileRequest(evt.Object, reqs)
- for req := range reqs {
- q.Add(req)
- }
-}
-
-// Generic implements EventHandler.
-func (e *EnqueueRequestForAncestor) Generic(evt event.GenericEvent, q workqueue.RateLimitingInterface) {
- reqs := map[reconcile.Request]empty{}
- e.getOwnerReconcileRequest(evt.Object, reqs)
- for req := range reqs {
- q.Add(req)
- }
-}
-
-// parseOwnerTypeGroupKind parses the OwnerType into a Group and Kind and caches the result. Returns false
-// if the OwnerType could not be parsed using the scheme.
-func (e *EnqueueRequestForAncestor) parseOwnerTypeGroupKind(scheme *runtime.Scheme) error {
- gk, err := e.parseTypeGroupKind(e.OwnerType, scheme)
- if err != nil {
- return err
- }
- // Cache the Group and Kind for the OwnerType
- e.groupKind = gk
- return nil
-}
-
-// parseInTypesGroupKind parses the InTypes into a Group and Kind and caches the result. Returns false
-// if the InTypes could not be parsed using the scheme.
-func (e *EnqueueRequestForAncestor) parseInTypesGroupKind(scheme *runtime.Scheme) error {
- if e.groupKind != nil {
- e.ancestorGroupKinds = append(e.ancestorGroupKinds, *e.groupKind)
- }
- for _, inType := range e.InTypes {
- gk, err := e.parseTypeGroupKind(inType, scheme)
- if err != nil {
- return err
- }
- // Cache the Group and Kind for the inType
- e.ancestorGroupKinds = append(e.ancestorGroupKinds, *gk)
- }
- return nil
-}
-
-func (e *EnqueueRequestForAncestor) parseTypeGroupKind(object runtime.Object, scheme *runtime.Scheme) (*schema.GroupKind, error) {
- // Get the kinds of the type
- kinds, _, err := scheme.ObjectKinds(object)
- if err != nil {
- log.Error(err, "Could not get ObjectKinds", "object", fmt.Sprintf("%T", object))
- return nil, err
- }
- // Expect only 1 kind. If there is more than one kind this is probably an edge case such as ListOptions.
- if len(kinds) != 1 {
- err := fmt.Errorf("expected exactly 1 kind for object %T, but found %s kinds", object, kinds)
- log.Error(nil, "expected exactly 1 kind for object", "object", fmt.Sprintf("%T", object), "kinds", kinds)
- return nil, err
- }
- return &schema.GroupKind{Group: kinds[0].Group, Kind: kinds[0].Kind}, nil
-}
-
-// getOwnerReconcileRequest looks at object and builds a map of reconcile.Request to reconcile
-// owners of object that match e.OwnerType.
-func (e *EnqueueRequestForAncestor) getOwnerReconcileRequest(obj client.Object, result map[reconcile.Request]empty) {
- // get the object by the ownerRef
- object, err := e.getSourceObject(obj)
- if err != nil {
- return
- }
-
- // find the root object up to UpToLevel
- scheme := *model.GetScheme()
- ctx := context.Background()
- ref, err := e.getOwnerUpTo(ctx, object, e.UpToLevel, scheme)
- if err != nil || ref == nil {
- return
- }
-
- // Parse the Group out of the OwnerReference to compare it to what was parsed out of the requested OwnerType
- refGV, err := schema.ParseGroupVersion(ref.APIVersion)
- if err != nil {
- log.Error(err, "Could not parse OwnerReference APIVersion",
- "api version", ref.APIVersion)
- return
- }
-
- // Compare the OwnerReference Group and Kind against the OwnerType Group and Kind specified by the user.
- // If the two match, create a Request for the objected referred to by
- // the OwnerReference. Use the Name from the OwnerReference and the Namespace from the
- // object in the event.
- if ref.Kind == e.groupKind.Kind && refGV.Group == e.groupKind.Group {
- // Match found - add a Request for the object referred to in the OwnerReference
- request := reconcile.Request{NamespacedName: types.NamespacedName{
- Name: ref.Name,
- }}
-
- // if owner is not namespaced then we should set the namespace to the empty
- mapping, err := e.mapper.RESTMapping(*e.groupKind, refGV.Version)
- if err != nil {
- log.Error(err, "Could not retrieve rest mapping", "kind", e.groupKind)
- return
- }
- if mapping.Scope.Name() != meta.RESTScopeNameRoot {
- request.Namespace = object.GetNamespace()
- }
-
- result[request] = empty{}
- }
-}
-
-func (e *EnqueueRequestForAncestor) getSourceObject(object client.Object) (client.Object, error) {
- eventObject, ok := object.(*corev1.Event)
- // return the object directly if it's not corev1.Event kind
- if !ok {
- return object, nil
- }
-
- objectRef := eventObject.InvolvedObject
- scheme := *model.GetScheme()
- // convert ObjectReference to OwnerReference
- ownerRef := metav1.OwnerReference{
- APIVersion: objectRef.APIVersion,
- Kind: objectRef.Kind,
- Name: objectRef.Name,
- UID: objectRef.UID,
- }
-
- ctx := context.Background()
- // get the object by the ownerRef
- sourceObject, err := e.getObjectByOwnerRef(ctx, objectRef.Namespace, ownerRef, scheme)
- if err != nil {
- return nil, err
- }
- return sourceObject, nil
-}
-
-// getOwnerUpTo gets the owner of object up to upToLevel.
-// E.g. If ReplicatedStateMachine creates the StatefulSet which then creates the Pod,
-// if the object is the Pod, then set upToLevel to 2 if you want to find the ReplicatedStateMachine.
-// Each level of ownership should be a controller-relationship (i.e. controller=true in ownerReferences).
-// nil return if no owner find in any level.
-func (e *EnqueueRequestForAncestor) getOwnerUpTo(ctx context.Context, object client.Object, upToLevel int, scheme runtime.Scheme) (*metav1.OwnerReference, error) {
- if upToLevel <= 0 {
- return nil, nil
- }
- if object == nil {
- return nil, nil
- }
- ownerRef := metav1.GetControllerOf(object)
- if ownerRef == nil {
- return nil, nil
- }
- if upToLevel == 1 {
- return ownerRef, nil
- }
- objectNew, err := e.getObjectByOwnerRef(ctx, object.GetNamespace(), *ownerRef, scheme)
- if err != nil {
- return nil, err
- }
- return e.getOwnerUpTo(ctx, objectNew, upToLevel-1, scheme)
-}
-
-func (e *EnqueueRequestForAncestor) getObjectByOwnerRef(ctx context.Context, ownerNameSpace string, ownerRef metav1.OwnerReference, scheme runtime.Scheme) (client.Object, error) {
- gv, err := schema.ParseGroupVersion(ownerRef.APIVersion)
- if err != nil {
- return nil, err
- }
- gvk := schema.GroupVersionKind{
- Group: gv.Group,
- Version: gv.Version,
- Kind: ownerRef.Kind,
- }
- if !e.inAncestorRange(gvk) {
- return nil, nil
- }
- objectRT, err := scheme.New(gvk)
- if err != nil {
- return nil, err
- }
- object, ok := objectRT.(client.Object)
- if !ok {
- return nil, errors.New("runtime object can't be converted to client object")
- }
- request := reconcile.Request{NamespacedName: types.NamespacedName{
- Name: ownerRef.Name,
- }}
- // if owner is not namespaced then we should set the namespace to the empty
- groupKind := schema.GroupKind{Group: gvk.Group, Kind: gvk.Kind}
- mapping, err := e.mapper.RESTMapping(groupKind, gvk.Version)
- if err != nil {
- return nil, err
- }
- if mapping.Scope.Name() != meta.RESTScopeNameRoot {
- request.Namespace = ownerNameSpace
- }
- if err := e.Client.Get(ctx, request.NamespacedName, object); err != nil {
- return nil, err
- }
- return object, nil
-}
-
-func (e *EnqueueRequestForAncestor) inAncestorRange(gvk schema.GroupVersionKind) bool {
- for _, groupKind := range e.ancestorGroupKinds {
- if gvk.Group == groupKind.Group && gvk.Kind == groupKind.Kind {
- return true
- }
- }
- return false
-}
-
-var _ inject.Scheme = &EnqueueRequestForAncestor{}
-
-// InjectScheme is called by the Controller to provide a singleton scheme to the EnqueueRequestForAncestor.
-func (e *EnqueueRequestForAncestor) InjectScheme(s *runtime.Scheme) error {
- if err := e.parseOwnerTypeGroupKind(s); err != nil {
- return err
- }
- return e.parseInTypesGroupKind(s)
-}
-
-var _ inject.Mapper = &EnqueueRequestForAncestor{}
-
-// InjectMapper is called by the Controller to provide the rest mapper used by the manager.
-func (e *EnqueueRequestForAncestor) InjectMapper(m meta.RESTMapper) error {
- e.mapper = m
- return nil
-}
diff --git a/internal/controller/rsm/enqueue_ancestor_test.go b/internal/controller/rsm/enqueue_ancestor_test.go
deleted file mode 100644
index 0f5576f1732..00000000000
--- a/internal/controller/rsm/enqueue_ancestor_test.go
+++ /dev/null
@@ -1,399 +0,0 @@
-/*
-Copyright (C) 2022-2023 ApeCloud Co., Ltd
-
-This file is part of KubeBlocks project
-
-This program is free software: you can redistribute it and/or modify
-it under the terms of the GNU Affero General Public License as published by
-the Free Software Foundation, either version 3 of the License, or
-(at your option) any later version.
-
-This program is distributed in the hope that it will be useful
-but WITHOUT ANY WARRANTY; without even the implied warranty of
-MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
-GNU Affero General Public License for more details.
-
-You should have received a copy of the GNU Affero General Public License
-along with this program. If not, see .
-*/
-
-package rsm
-
-import (
- "context"
- "fmt"
-
- . "github.com/onsi/ginkgo/v2"
- . "github.com/onsi/gomega"
-
- "github.com/golang/mock/gomock"
- appsv1 "k8s.io/api/apps/v1"
- corev1 "k8s.io/api/core/v1"
- "k8s.io/apimachinery/pkg/api/meta"
- metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
- "k8s.io/apimachinery/pkg/runtime"
- "k8s.io/apimachinery/pkg/runtime/schema"
- "k8s.io/apimachinery/pkg/types"
- "k8s.io/client-go/util/workqueue"
- "sigs.k8s.io/controller-runtime/pkg/client"
- "sigs.k8s.io/controller-runtime/pkg/event"
- "sigs.k8s.io/controller-runtime/pkg/reconcile"
-
- workloads "github.com/apecloud/kubeblocks/apis/workloads/v1alpha1"
- "github.com/apecloud/kubeblocks/internal/controller/builder"
- "github.com/apecloud/kubeblocks/internal/controller/model"
-)
-
-func init() {
- model.AddScheme(workloads.AddToScheme)
-}
-
-var _ = Describe("enqueue ancestor", func() {
- scheme := model.GetScheme()
- var handler *EnqueueRequestForAncestor
-
- buildAncestorTree := func() (*workloads.ReplicatedStateMachine, *appsv1.StatefulSet, *corev1.Pod) {
- ancestorL2APIVersion := "workloads.kubeblocks.io/v1alpha1"
- ancestorL2Kind := "ReplicatedStateMachine"
- ancestorL2Name := "ancestor-level-2"
- ancestorL1APIVersion := "apps/v1"
- ancestorL1Kind := "StatefulSet"
- ancestorL1Name := "ancestor-level-1"
- objectName := ancestorL1Name + "-0"
-
- ancestorLevel2 := builder.NewReplicatedStateMachineBuilder(namespace, ancestorL2Name).GetObject()
- ancestorLevel2.APIVersion = ancestorL2APIVersion
- ancestorLevel2.Kind = ancestorL2Kind
- ancestorLevel1 := builder.NewStatefulSetBuilder(namespace, ancestorL1Name).
- SetOwnerReferences(ancestorL2APIVersion, ancestorL2Kind, ancestorLevel2).
- GetObject()
- ancestorLevel1.APIVersion = ancestorL1APIVersion
- ancestorLevel1.Kind = ancestorL1Kind
- object := builder.NewPodBuilder(namespace, objectName).
- SetOwnerReferences(ancestorL1APIVersion, ancestorL1Kind, ancestorLevel1).
- GetObject()
-
- return ancestorLevel2, ancestorLevel1, object
- }
-
- BeforeEach(func() {
- handler = &EnqueueRequestForAncestor{
- Client: k8sMock,
- OwnerType: &workloads.ReplicatedStateMachine{},
- UpToLevel: 2,
- InTypes: []runtime.Object{&appsv1.StatefulSet{}},
- }
- })
-
- Context("parseOwnerTypeGroupKind", func() {
- It("should work well", func() {
- Expect(handler.parseOwnerTypeGroupKind(scheme)).Should(Succeed())
- Expect(handler.groupKind.Group).Should(Equal("workloads.kubeblocks.io"))
- Expect(handler.groupKind.Kind).Should(Equal("ReplicatedStateMachine"))
- })
- })
-
- Context("parseInTypesGroupKind", func() {
- It("should work well", func() {
- Expect(handler.parseInTypesGroupKind(scheme)).Should(Succeed())
- Expect(handler.ancestorGroupKinds).Should(HaveLen(1))
- Expect(handler.ancestorGroupKinds[0].Group).Should(Equal("apps"))
- Expect(handler.ancestorGroupKinds[0].Kind).Should(Equal("StatefulSet"))
- })
- })
-
- Context("getObjectByOwnerRef", func() {
- BeforeEach(func() {
- Expect(handler.InjectScheme(scheme)).Should(Succeed())
- Expect(handler.InjectMapper(newFakeMapper())).Should(Succeed())
- })
-
- It("should return err if groupVersion parsing error", func() {
- wrongAPIVersion := "wrong/group/version"
- ownerRef := metav1.OwnerReference{
- APIVersion: wrongAPIVersion,
- }
- _, err := handler.getObjectByOwnerRef(ctx, namespace, ownerRef, *scheme)
- Expect(err).ShouldNot(BeNil())
- Expect(err.Error()).Should(ContainSubstring(wrongAPIVersion))
- })
-
- It("should return nil if ancestor's type out of range", func() {
- ownerRef := metav1.OwnerReference{
- APIVersion: "apps/v1",
- Kind: "Deployment",
- Name: "foo",
- UID: "bar",
- }
- object, err := handler.getObjectByOwnerRef(ctx, namespace, ownerRef, *scheme)
- Expect(err).Should(BeNil())
- Expect(object).Should(BeNil())
- })
-
- It("should return the owner object", func() {
- ownerName := "foo"
- ownerUID := types.UID("bar")
- ownerRef := metav1.OwnerReference{
- APIVersion: "apps/v1",
- Kind: "StatefulSet",
- Name: ownerName,
- UID: ownerUID,
- }
- k8sMock.EXPECT().
- Get(gomock.Any(), gomock.Any(), &appsv1.StatefulSet{}, gomock.Any()).
- DoAndReturn(func(_ context.Context, _ client.ObjectKey, obj *appsv1.StatefulSet, _ ...client.ListOption) error {
- obj.Name = ownerName
- obj.UID = ownerUID
- return nil
- }).Times(1)
- object, err := handler.getObjectByOwnerRef(ctx, namespace, ownerRef, *scheme)
- Expect(err).Should(BeNil())
- Expect(object).ShouldNot(BeNil())
- Expect(object.GetName()).Should(Equal(ownerName))
- Expect(object.GetUID()).Should(Equal(ownerUID))
- })
- })
-
- Context("getOwnerUpTo", func() {
- BeforeEach(func() {
- Expect(handler.InjectScheme(scheme)).Should(Succeed())
- Expect(handler.InjectMapper(newFakeMapper())).Should(Succeed())
- })
-
- It("should work well", func() {
- By("set upToLevel to 0")
- ownerRef, err := handler.getOwnerUpTo(ctx, nil, 0, *scheme)
- Expect(err).Should(BeNil())
- Expect(ownerRef).Should(BeNil())
-
- By("set object to nil")
- ownerRef, err = handler.getOwnerUpTo(ctx, nil, handler.UpToLevel, *scheme)
- Expect(err).Should(BeNil())
- Expect(ownerRef).Should(BeNil())
-
- By("builder ancestor tree")
- ancestorLevel2, ancestorLevel1, object := buildAncestorTree()
-
- By("set upToLevel to 1")
- ownerRef, err = handler.getOwnerUpTo(ctx, object, 1, *scheme)
- Expect(err).Should(BeNil())
- Expect(ownerRef).ShouldNot(BeNil())
- Expect(ownerRef.APIVersion).Should(Equal(ancestorLevel1.APIVersion))
- Expect(ownerRef.Kind).Should(Equal(ancestorLevel1.Kind))
- Expect(ownerRef.Name).Should(Equal(ancestorLevel1.Name))
- Expect(ownerRef.UID).Should(Equal(ancestorLevel1.UID))
-
- By("set upToLevel to 2")
- k8sMock.EXPECT().
- Get(gomock.Any(), gomock.Any(), &appsv1.StatefulSet{}, gomock.Any()).
- DoAndReturn(func(_ context.Context, objKey client.ObjectKey, sts *appsv1.StatefulSet, _ ...client.ListOptions) error {
- sts.Namespace = objKey.Namespace
- sts.Name = objKey.Name
- sts.OwnerReferences = ancestorLevel1.OwnerReferences
- return nil
- }).Times(1)
- ownerRef, err = handler.getOwnerUpTo(ctx, object, handler.UpToLevel, *scheme)
- Expect(err).Should(BeNil())
- Expect(ownerRef).ShouldNot(BeNil())
- Expect(ownerRef.APIVersion).Should(Equal(ancestorLevel2.APIVersion))
- Expect(ownerRef.Kind).Should(Equal(ancestorLevel2.Kind))
- Expect(ownerRef.Name).Should(Equal(ancestorLevel2.Name))
- Expect(ownerRef.UID).Should(Equal(ancestorLevel2.UID))
- })
- })
-
- Context("getSourceObject", func() {
- BeforeEach(func() {
- Expect(handler.InjectScheme(scheme)).Should(Succeed())
- Expect(handler.InjectMapper(newFakeMapper())).Should(Succeed())
- })
-
- It("should work well", func() {
- By("build a non-event object")
- name := "foo"
- uid := types.UID("bar")
- object1 := builder.NewPodBuilder(namespace, name).SetUID(uid).GetObject()
- objectSrc1, err := handler.getSourceObject(object1)
- Expect(err).Should(BeNil())
- Expect(objectSrc1).Should(Equal(object1))
-
- By("build an event object")
- handler.InTypes = append(handler.InTypes, &corev1.Pod{})
- Expect(handler.InjectScheme(scheme)).Should(Succeed())
- objectRef := corev1.ObjectReference{
- APIVersion: "v1",
- Kind: "Pod",
- Namespace: namespace,
- Name: object1.Name,
- UID: object1.UID,
- }
- object2 := builder.NewEventBuilder(namespace, "foo").
- SetInvolvedObject(objectRef).
- GetObject()
- k8sMock.EXPECT().
- Get(gomock.Any(), gomock.Any(), &corev1.Pod{}, gomock.Any()).
- DoAndReturn(func(_ context.Context, objKey client.ObjectKey, obj *corev1.Pod, _ ...client.ListOptions) error {
- obj.Name = objKey.Name
- obj.Namespace = objKey.Namespace
- obj.UID = objectRef.UID
- return nil
- }).Times(1)
- objectSrc2, err := handler.getSourceObject(object2)
- Expect(err).Should(BeNil())
- Expect(objectSrc2).ShouldNot(BeNil())
- Expect(objectSrc2.GetName()).Should(Equal(object1.Name))
- Expect(objectSrc2.GetNamespace()).Should(Equal(object1.Namespace))
- Expect(objectSrc2.GetUID()).Should(Equal(object1.UID))
- })
- })
-
- Context("getOwnerReconcileRequest", func() {
- BeforeEach(func() {
- Expect(handler.InjectScheme(scheme)).Should(Succeed())
- Expect(handler.InjectMapper(newFakeMapper())).Should(Succeed())
- })
-
- It("should work well", func() {
- By("build ancestor tree")
- ancestorLevel2, ancestorLevel1, object := buildAncestorTree()
-
- k8sMock.EXPECT().
- Get(gomock.Any(), gomock.Any(), &appsv1.StatefulSet{}, gomock.Any()).
- DoAndReturn(func(_ context.Context, objKey client.ObjectKey, sts *appsv1.StatefulSet, _ ...client.ListOptions) error {
- sts.Namespace = objKey.Namespace
- sts.Name = objKey.Name
- sts.OwnerReferences = ancestorLevel1.OwnerReferences
- return nil
- }).Times(1)
-
- By("get object with ancestors")
- result := make(map[reconcile.Request]empty)
- handler.getOwnerReconcileRequest(object, result)
- Expect(result).Should(HaveLen(1))
- for request := range result {
- Expect(request.Namespace).Should(Equal(ancestorLevel2.Namespace))
- Expect(request.Name).Should(Equal(ancestorLevel2.Name))
- }
-
- By("set obj not exist")
- wrongAPIVersion := "wrong/api/version"
- object.OwnerReferences[0].APIVersion = wrongAPIVersion
- result = make(map[reconcile.Request]empty)
- handler.getOwnerReconcileRequest(object, result)
- Expect(result).Should(HaveLen(0))
-
- By("set level 1 ancestor's owner not exist")
- object.OwnerReferences[0].APIVersion = ancestorLevel1.APIVersion
- k8sMock.EXPECT().
- Get(gomock.Any(), gomock.Any(), &appsv1.StatefulSet{}, gomock.Any()).
- DoAndReturn(func(_ context.Context, objKey client.ObjectKey, sts *appsv1.StatefulSet, _ ...client.ListOptions) error {
- sts.Namespace = objKey.Namespace
- sts.Name = objKey.Name
- return nil
- }).Times(1)
- result = make(map[reconcile.Request]empty)
- handler.getOwnerReconcileRequest(object, result)
- Expect(result).Should(HaveLen(0))
- })
- })
-
- Context("handler interface", func() {
- BeforeEach(func() {
- Expect(handler.InjectScheme(scheme)).Should(Succeed())
- Expect(handler.InjectMapper(newFakeMapper())).Should(Succeed())
- })
-
- It("should work well", func() {
- By("build events and queue")
- queue := workqueue.NewNamedRateLimitingQueue(workqueue.DefaultControllerRateLimiter(), "enqueue-ancestor-test")
- ancestorLevel2, ancestorLevel1, object := buildAncestorTree()
- createEvent := event.CreateEvent{Object: object}
- updateEvent := event.UpdateEvent{ObjectOld: object, ObjectNew: object}
- deleteEvent := event.DeleteEvent{Object: object}
- genericEvent := event.GenericEvent{Object: object}
-
- cases := []struct {
- name string
- testFunc func()
- getTimes int
- }{
- {
- name: "Create",
- testFunc: func() { handler.Create(createEvent, queue) },
- getTimes: 1,
- },
- {
- name: "Update",
- testFunc: func() { handler.Update(updateEvent, queue) },
- getTimes: 2,
- },
- {
- name: "Delete",
- testFunc: func() { handler.Delete(deleteEvent, queue) },
- getTimes: 1,
- },
- {
- name: "Generic",
- testFunc: func() { handler.Generic(genericEvent, queue) },
- getTimes: 1,
- },
- }
- for _, c := range cases {
- By(fmt.Sprintf("test %s interface", c.name))
- k8sMock.EXPECT().
- Get(gomock.Any(), gomock.Any(), &appsv1.StatefulSet{}, gomock.Any()).
- DoAndReturn(func(_ context.Context, objKey client.ObjectKey, sts *appsv1.StatefulSet, _ ...client.ListOptions) error {
- sts.Namespace = objKey.Namespace
- sts.Name = objKey.Name
- sts.OwnerReferences = ancestorLevel1.OwnerReferences
- return nil
- }).Times(c.getTimes)
- c.testFunc()
- item, shutdown := queue.Get()
- Expect(shutdown).Should(BeFalse())
- request, ok := item.(reconcile.Request)
- Expect(ok).Should(BeTrue())
- Expect(request.Namespace).Should(Equal(ancestorLevel2.Namespace))
- Expect(request.Name).Should(Equal(ancestorLevel2.Name))
- queue.Done(item)
- queue.Forget(item)
- }
-
- queue.ShutDown()
- })
- })
-})
-
-type fakeMapper struct{}
-
-func (f *fakeMapper) KindFor(resource schema.GroupVersionResource) (schema.GroupVersionKind, error) {
- return schema.GroupVersionKind{}, nil
-}
-
-func (f *fakeMapper) KindsFor(resource schema.GroupVersionResource) ([]schema.GroupVersionKind, error) {
- return nil, nil
-}
-
-func (f *fakeMapper) ResourceFor(input schema.GroupVersionResource) (schema.GroupVersionResource, error) {
- return schema.GroupVersionResource{}, nil
-}
-
-func (f *fakeMapper) ResourcesFor(input schema.GroupVersionResource) ([]schema.GroupVersionResource, error) {
- return nil, nil
-}
-
-func (f *fakeMapper) RESTMapping(gk schema.GroupKind, versions ...string) (*meta.RESTMapping, error) {
- return &meta.RESTMapping{Scope: meta.RESTScopeNamespace}, nil
-}
-
-func (f *fakeMapper) RESTMappings(gk schema.GroupKind, versions ...string) ([]*meta.RESTMapping, error) {
- return nil, nil
-}
-
-func (f *fakeMapper) ResourceSingularizer(resource string) (singular string, err error) {
- return "", nil
-}
-
-func newFakeMapper() meta.RESTMapper {
- return &fakeMapper{}
-}
diff --git a/internal/controller/rsm/pod_role_event_handler.go b/internal/controller/rsm/pod_role_event_handler.go
index 6af517052cd..36ce5ae973a 100644
--- a/internal/controller/rsm/pod_role_event_handler.go
+++ b/internal/controller/rsm/pod_role_event_handler.go
@@ -24,6 +24,7 @@ import (
"fmt"
"regexp"
"strings"
+ "time"
corev1 "k8s.io/api/core/v1"
"k8s.io/apimachinery/pkg/types"
@@ -31,17 +32,21 @@ import (
"sigs.k8s.io/controller-runtime/pkg/client"
workloads "github.com/apecloud/kubeblocks/apis/workloads/v1alpha1"
+ "github.com/apecloud/kubeblocks/internal/common"
+ "github.com/apecloud/kubeblocks/internal/constant"
intctrlutil "github.com/apecloud/kubeblocks/internal/controllerutil"
)
-// TODO(free6om): dedup copied funcs from event_controllers.go
-// TODO(free6om): refactor event_controller.go as it should NOT import controllers/apps/component/*
-
type PodRoleEventHandler struct{}
// probeEventType defines the type of probe event.
type probeEventType string
+const (
+ successEvent = "Success"
+ roleChangedEvent = "roleChanged"
+)
+
type probeMessage struct {
Event probeEventType `json:"event,omitempty"`
Message string `json:"message,omitempty"`
@@ -57,7 +62,10 @@ const (
var roleMessageRegex = regexp.MustCompile(`Readiness probe failed: .*({.*})`)
func (h *PodRoleEventHandler) Handle(cli client.Client, reqCtx intctrlutil.RequestCtx, recorder record.EventRecorder, event *corev1.Event) error {
- if event.InvolvedObject.FieldPath != roleProbeEventFieldPath {
+ if event.InvolvedObject.FieldPath != readinessProbeEventFieldPath &&
+ event.InvolvedObject.FieldPath != directAPIServerEventFieldPath &&
+ event.InvolvedObject.FieldPath != legacyEventFieldPath &&
+ event.Reason != checkRoleEventReason {
return nil
}
var (
@@ -93,50 +101,95 @@ func handleRoleChangedEvent(cli client.Client, reqCtx intctrlutil.RequestCtx, re
}
// if probe event operation is not impl, check role failed or role invalid, ignore it
- if message.Event != "Success" {
+ if message.Event != successEvent && message.Event != roleChangedEvent {
reqCtx.Log.Info("probe event failed", "message", message.Message)
return "", nil
}
role := strings.ToLower(message.Role)
- podName := types.NamespacedName{
- Namespace: event.InvolvedObject.Namespace,
- Name: event.InvolvedObject.Name,
- }
- // get pod
- pod := &corev1.Pod{}
- if err := cli.Get(reqCtx.Ctx, podName, pod); err != nil {
- return role, err
+ snapshot := parseGlobalRoleSnapshot(role, event)
+ for _, pair := range snapshot.PodRoleNamePairs {
+ podName := types.NamespacedName{
+ Namespace: event.InvolvedObject.Namespace,
+ Name: pair.PodName,
+ }
+ // get pod
+ pod := &corev1.Pod{}
+ if err := cli.Get(reqCtx.Ctx, podName, pod); err != nil {
+ return pair.RoleName, err
+ }
+ // event belongs to old pod with the same name, ignore it
+ if pod.Name == pair.PodName && pod.UID != event.InvolvedObject.UID {
+ return pair.RoleName, nil
+ }
+
+ // compare the version of the current role snapshot with the last version recorded in the pod annotation,
+ // stale role snapshot will be ignored.
+ lastSnapshotVersion, ok := pod.Annotations[constant.LastRoleSnapshotVersionAnnotationKey]
+ if ok {
+
+ if snapshot.Version <= lastSnapshotVersion {
+ reqCtx.Log.Info("stale role snapshot received, ignore it", "snapshot", snapshot)
+ return pair.RoleName, nil
+ }
+ }
+
+ name, _ := intctrlutil.GetParentNameAndOrdinal(pod)
+ rsm := &workloads.ReplicatedStateMachine{}
+ if err := cli.Get(reqCtx.Ctx, types.NamespacedName{Namespace: pod.Namespace, Name: name}, rsm); err != nil {
+ return "", err
+ }
+ reqCtx.Log.V(1).Info("handle role change event", "pod", pod.Name, "role", role, "originalRole", message.OriginalRole)
+
+ if err := updatePodRoleLabel(cli, reqCtx, *rsm, pod, pair.RoleName, snapshot.Version); err != nil {
+ return "", err
+ }
}
- // event belongs to old pod with the same name, ignore it
- if pod.UID != event.InvolvedObject.UID {
- return role, nil
+ return role, nil
+}
+
+func parseGlobalRoleSnapshot(role string, event *corev1.Event) *common.GlobalRoleSnapshot {
+ snapshot := &common.GlobalRoleSnapshot{}
+ if err := json.Unmarshal([]byte(role), snapshot); err == nil {
+ return snapshot
}
- name, _ := intctrlutil.GetParentNameAndOrdinal(pod)
- rsm := &workloads.ReplicatedStateMachine{}
- if err := cli.Get(reqCtx.Ctx, types.NamespacedName{Namespace: pod.Namespace, Name: name}, rsm); err != nil {
- return "", err
+ snapshot.Version = event.EventTime.Time.Format(time.RFC3339Nano)
+ pair := common.PodRoleNamePair{
+ PodName: event.InvolvedObject.Name,
+ RoleName: role,
}
- reqCtx.Log.V(1).Info("handle role change event", "pod", pod.Name, "role", role, "originalRole", message.OriginalRole)
-
- return role, updatePodRoleLabel(cli, reqCtx, *rsm, pod, role)
+ snapshot.PodRoleNamePairs = append(snapshot.PodRoleNamePairs, pair)
+ return snapshot
}
// parseProbeEventMessage parses probe event message.
func parseProbeEventMessage(reqCtx intctrlutil.RequestCtx, event *corev1.Event) *probeMessage {
message := &probeMessage{}
- matches := roleMessageRegex.FindStringSubmatch(event.Message)
- if len(matches) != 2 {
- reqCtx.Log.Info("parser Readiness probe event message failed", "message", event.Message)
- return nil
+ tryUnmarshalDirectAPIServerEvent := func() error {
+ return json.Unmarshal([]byte(event.Message), message)
}
- msg := matches[1]
- err := json.Unmarshal([]byte(msg), message)
- if err != nil {
- // not role related message, ignore it
- reqCtx.Log.Info("not role message", "message", event.Message, "error", err)
+ tryUnmarshalReadinessProbeEvent := func() error {
+ matches := roleMessageRegex.FindStringSubmatch(event.Message)
+ if len(matches) != 2 {
+ reqCtx.Log.Info("parser Readiness probe event message failed", "message", event.Message)
+ return fmt.Errorf("parser Readiness probe event message failed: %s", event.Message)
+ }
+ msg := matches[1]
+ err := json.Unmarshal([]byte(msg), message)
+ if err != nil {
+ // not role related message, ignore it
+ reqCtx.Log.Info("not role message", "message", event.Message, "error", err)
+ return err
+ }
return nil
}
- return message
+
+ if err := tryUnmarshalDirectAPIServerEvent(); err == nil {
+ return message
+ }
+ if err := tryUnmarshalReadinessProbeEvent(); err == nil {
+ return message
+ }
+ return nil
}
diff --git a/internal/controller/rsm/pod_role_event_handler_test.go b/internal/controller/rsm/pod_role_event_handler_test.go
index f062077c5de..abfee5d5ce7 100644
--- a/internal/controller/rsm/pod_role_event_handler_test.go
+++ b/internal/controller/rsm/pod_role_event_handler_test.go
@@ -52,7 +52,7 @@ var _ = Describe("pod role label event handler test", func() {
Namespace: pod.Namespace,
Name: pod.Name,
UID: pod.UID,
- FieldPath: roleProbeEventFieldPath,
+ FieldPath: readinessProbeEventFieldPath,
}
role := workloads.ReplicaRole{
Name: "leader",
diff --git a/internal/controller/rsm/transformer_object_generation.go b/internal/controller/rsm/transformer_object_generation.go
index 96a7f28a117..421f2d8788e 100644
--- a/internal/controller/rsm/transformer_object_generation.go
+++ b/internal/controller/rsm/transformer_object_generation.go
@@ -27,6 +27,7 @@ import (
"strings"
"golang.org/x/exp/maps"
+ "golang.org/x/exp/slices"
apps "k8s.io/api/apps/v1"
corev1 "k8s.io/api/core/v1"
"k8s.io/apimachinery/pkg/util/intstr"
@@ -401,7 +402,6 @@ func injectRoleProbeAgentContainer(rsm workloads.ReplicatedStateMachine, templat
if probeDaemonPort == 0 {
probeDaemonPort = defaultRoleProbeDaemonPort
}
- roleProbeURI := fmt.Sprintf(roleProbeURIFormat, strconv.Itoa(probeDaemonPort))
env := credentialEnv
env = append(env,
corev1.EnvVar{
@@ -437,40 +437,124 @@ func injectRoleProbeAgentContainer(rsm workloads.ReplicatedStateMachine, templat
})
}
- // build container
- container := corev1.Container{
- Name: roleProbeName,
- Image: image,
- ImagePullPolicy: "IfNotPresent",
- Command: []string{
- "role-agent",
- "--port", strconv.Itoa(probeDaemonPort),
+ // inject role update mechanism env
+ env = append(env,
+ corev1.EnvVar{
+ Name: RoleUpdateMechanismVarName,
+ Value: string(roleProbe.RoleUpdateMechanism),
+ })
+
+ // inject role probe timeout env
+ env = append(env,
+ corev1.EnvVar{
+ Name: roleProbeTimeoutVarName,
+ Value: strconv.Itoa(int(roleProbe.TimeoutSeconds)),
+ })
+
+ // lorry related envs
+ env = append(env,
+ corev1.EnvVar{
+ Name: constant.KBEnvPodName,
+ ValueFrom: &corev1.EnvVarSource{
+ FieldRef: &corev1.ObjectFieldSelector{
+ FieldPath: "metadata.name",
+ },
+ },
},
- Ports: []corev1.ContainerPort{{
- ContainerPort: int32(probeDaemonPort),
- Name: roleProbeName,
- Protocol: "TCP",
- }},
- ReadinessProbe: &corev1.Probe{
- ProbeHandler: corev1.ProbeHandler{
- Exec: &corev1.ExecAction{
- Command: []string{
- "/bin/grpc_health_probe",
- roleProbeURI,
- },
+ corev1.EnvVar{
+ Name: constant.KBEnvNamespace,
+ ValueFrom: &corev1.EnvVarSource{
+ FieldRef: &corev1.ObjectFieldSelector{
+ FieldPath: "metadata.namespace",
},
},
- InitialDelaySeconds: roleProbe.InitialDelaySeconds,
- TimeoutSeconds: roleProbe.TimeoutSeconds,
- PeriodSeconds: roleProbe.PeriodSeconds,
- SuccessThreshold: roleProbe.SuccessThreshold,
- FailureThreshold: roleProbe.FailureThreshold,
},
- Env: env,
+ corev1.EnvVar{
+ Name: constant.KBEnvPodUID,
+ ValueFrom: &corev1.EnvVarSource{
+ FieldRef: &corev1.ObjectFieldSelector{
+ FieldPath: "metadata.uid",
+ },
+ },
+ },
+ corev1.EnvVar{
+ Name: constant.KBEnvNodeName,
+ ValueFrom: &corev1.EnvVarSource{
+ FieldRef: &corev1.ObjectFieldSelector{
+ FieldPath: "spec.nodeName",
+ },
+ },
+ },
+ )
+
+ readinessProbe := &corev1.Probe{
+ ProbeHandler: corev1.ProbeHandler{
+ HTTPGet: &corev1.HTTPGetAction{
+ Path: roleProbeURI,
+ Port: intstr.FromInt(probeDaemonPort),
+ },
+ },
+ InitialDelaySeconds: roleProbe.InitialDelaySeconds,
+ TimeoutSeconds: roleProbe.TimeoutSeconds,
+ PeriodSeconds: roleProbe.PeriodSeconds,
+ SuccessThreshold: roleProbe.SuccessThreshold,
+ FailureThreshold: roleProbe.FailureThreshold,
}
+ tryToGetRoleProbeContainer := func() *corev1.Container {
+ for i, container := range template.Spec.Containers {
+ if container.Image != image {
+ continue
+ }
+ if len(container.Command) == 0 || container.Command[0] != roleProbeBinaryName {
+ continue
+ }
+ if container.ReadinessProbe != nil {
+ continue
+ }
+ // if all the above conditions satisfied, container that can do the role probe job found
+ return &template.Spec.Containers[i]
+ }
+ return nil
+ }
+ // if role probe container exists, update the readiness probe, env and serving container port
+ if container := tryToGetRoleProbeContainer(); container != nil {
+ // presume the first port is the http port.
+ // this is an easily broken contract between rsm controller and cluster controller.
+ // TODO(free6om): design a better way to do this after Lorry-WeSyncer separation done
+ readinessProbe.HTTPGet.Port = intstr.FromInt(int(container.Ports[0].ContainerPort))
+ container.ReadinessProbe = readinessProbe
+ for _, e := range env {
+ if slices.IndexFunc(container.Env, func(v corev1.EnvVar) bool {
+ return v.Name == e.Name
+ }) >= 0 {
+ continue
+ }
+ container.Env = append(container.Env, e)
+ }
+ return
+ }
+
+ // if role probe container doesn't exist, create a new one
+ // build container
+ container := builder.NewContainerBuilder(roleProbeContainerName).
+ SetImage(image).
+ SetImagePullPolicy(corev1.PullIfNotPresent).
+ AddCommands([]string{
+ roleProbeBinaryName,
+ "--port", strconv.Itoa(probeDaemonPort),
+ }...).
+ AddEnv(env...).
+ AddPorts(corev1.ContainerPort{
+ ContainerPort: int32(probeDaemonPort),
+ Name: roleProbeContainerName,
+ Protocol: "TCP",
+ }).
+ SetReadinessProbe(*readinessProbe).
+ GetObject()
+
// inject role probe container
- template.Spec.Containers = append(template.Spec.Containers, container)
+ template.Spec.Containers = append(template.Spec.Containers, *container)
}
func injectProbeActionContainer(rsm workloads.ReplicatedStateMachine, template *corev1.PodTemplateSpec, actionSvcPorts []int32, credentialEnv []corev1.EnvVar) {
diff --git a/internal/controller/rsm/types.go b/internal/controller/rsm/types.go
index 1e769aed740..9912c48143b 100644
--- a/internal/controller/rsm/types.go
+++ b/internal/controller/rsm/types.go
@@ -69,26 +69,33 @@ const (
jobScenarioMembership = "membership-reconfiguration"
jobScenarioUpdate = "pod-update"
- roleProbeName = "role-observe"
- roleAgentVolumeName = "role-agent"
- roleAgentInstallerName = "role-agent-installer"
- roleAgentVolumeMountPath = "/role-probe"
- roleAgentName = "agent"
- shell2httpImage = "msoap/shell2http:1.16.0"
- shell2httpBinaryPath = "/app/shell2http"
- shell2httpServePath = "/role"
- defaultRoleProbeAgentImage = "apecloud/kubeblocks-role-agent:latest"
- defaultRoleProbeDaemonPort = 7373
- roleProbeURIFormat = "-addr=localhost:%s"
- defaultActionImage = "busybox:latest"
- usernameCredentialVarName = "KB_RSM_USERNAME"
- passwordCredentialVarName = "KB_RSM_PASSWORD"
- servicePortVarName = "KB_RSM_SERVICE_PORT"
- actionSvcListVarName = "KB_RSM_ACTION_SVC_LIST"
- leaderHostVarName = "KB_RSM_LEADER_HOST"
- targetHostVarName = "KB_RSM_TARGET_HOST"
- roleProbeEventFieldPath = "spec.containers{" + roleProbeName + "}"
- actionSvcPortBase = int32(36500)
+ roleProbeContainerName = "kb-role-probe"
+ roleProbeBinaryName = "lorry"
+ roleAgentVolumeName = "role-agent"
+ roleAgentInstallerName = "role-agent-installer"
+ roleAgentVolumeMountPath = "/role-probe"
+ roleAgentName = "agent"
+ shell2httpImage = "msoap/shell2http:1.16.0"
+ shell2httpBinaryPath = "/app/shell2http"
+ shell2httpServePath = "/role"
+ defaultRoleProbeAgentImage = "apecloud/kubeblocks-tools:latest"
+ defaultRoleProbeDaemonPort = 7373
+ roleProbeURI = "/v1.0/bindings/custom?operation=checkRole"
+ defaultActionImage = "busybox:latest"
+ usernameCredentialVarName = "KB_RSM_USERNAME"
+ passwordCredentialVarName = "KB_RSM_PASSWORD"
+ servicePortVarName = "KB_RSM_SERVICE_PORT"
+ actionSvcListVarName = "KB_RSM_ACTION_SVC_LIST"
+ leaderHostVarName = "KB_RSM_LEADER_HOST"
+ targetHostVarName = "KB_RSM_TARGET_HOST"
+ RoleUpdateMechanismVarName = "KB_RSM_ROLE_UPDATE_MECHANISM"
+ roleProbeTimeoutVarName = "KB_RSM_ROLE_PROBE_TIMEOUT"
+ directAPIServerEventFieldPath = "spec.containers{sqlchannel}"
+ readinessProbeEventFieldPath = "spec.containers{" + roleProbeContainerName + "}"
+ legacyEventFieldPath = "spec.containers{kb-checkrole}"
+ checkRoleEventReason = "checkRole"
+
+ actionSvcPortBase = int32(36500)
)
type rsmTransformContext struct {
diff --git a/internal/controller/rsm/update_plan.go b/internal/controller/rsm/update_plan.go
index 16c2ee4abc5..0a6b712411d 100644
--- a/internal/controller/rsm/update_plan.go
+++ b/internal/controller/rsm/update_plan.go
@@ -94,8 +94,8 @@ func (p *realUpdatePlan) build() {
return
}
- rolePriorityMap := composeRolePriorityMap(p.rsm)
- sortPods(p.pods, rolePriorityMap, false)
+ rolePriorityMap := ComposeRolePriorityMap(p.rsm.Spec.Roles)
+ SortPods(p.pods, rolePriorityMap, false)
// generate plan by MemberUpdateStrategy
switch *p.rsm.Spec.MemberUpdateStrategy {
diff --git a/internal/controller/rsm/utils.go b/internal/controller/rsm/utils.go
index 15cdb802c8e..7efc55b7e03 100644
--- a/internal/controller/rsm/utils.go
+++ b/internal/controller/rsm/utils.go
@@ -63,10 +63,10 @@ const (
var podNameRegex = regexp.MustCompile(`(.*)-([0-9]+)$`)
-// sortPods sorts pods by their role priority
+// SortPods sorts pods by their role priority
// e.g.: unknown -> empty -> learner -> follower1 -> follower2 -> leader, with follower1.Name < follower2.Name
// reverse it if reverse==true
-func sortPods(pods []corev1.Pod, rolePriorityMap map[string]int, reverse bool) {
+func SortPods(pods []corev1.Pod, rolePriorityMap map[string]int, reverse bool) {
getRoleFunc := func(i int) string {
return getRoleName(pods[i])
}
@@ -88,11 +88,12 @@ func sortMembersStatus(membersStatus []workloads.MemberStatus, rolePriorityMap m
sortMembers(membersStatus, rolePriorityMap, getRoleFunc, getOrdinalFunc, true)
}
-func sortMembers[T any](membersStatus []T,
+// sortMembers sorts items by role priority and pod ordinal.
+func sortMembers[T any](items []T,
rolePriorityMap map[string]int,
getRoleFunc getRole, getOrdinalFunc getOrdinal,
reverse bool) {
- sort.SliceStable(membersStatus, func(i, j int) bool {
+ sort.SliceStable(items, func(i, j int) bool {
if reverse {
i, j = j, i
}
@@ -107,11 +108,11 @@ func sortMembers[T any](membersStatus []T,
})
}
-// composeRolePriorityMap generates a priority map based on roles.
-func composeRolePriorityMap(rsm workloads.ReplicatedStateMachine) map[string]int {
+// ComposeRolePriorityMap generates a priority map based on roles.
+func ComposeRolePriorityMap(roles []workloads.ReplicaRole) map[string]int {
rolePriorityMap := make(map[string]int, 0)
rolePriorityMap[""] = emptyPriority
- for _, role := range rsm.Spec.Roles {
+ for _, role := range roles {
roleName := strings.ToLower(role.Name)
switch {
case role.IsLeader:
@@ -134,10 +135,8 @@ func composeRolePriorityMap(rsm workloads.ReplicatedStateMachine) map[string]int
}
// updatePodRoleLabel updates pod role label when internal container role changed
-func updatePodRoleLabel(cli client.Client,
- reqCtx intctrlutil.RequestCtx,
- rsm workloads.ReplicatedStateMachine,
- pod *corev1.Pod, roleName string) error {
+func updatePodRoleLabel(cli client.Client, reqCtx intctrlutil.RequestCtx,
+ rsm workloads.ReplicatedStateMachine, pod *corev1.Pod, roleName string, version string) error {
ctx := reqCtx.Ctx
roleMap := composeRoleMap(rsm)
// role not defined in CR, ignore it
@@ -154,6 +153,11 @@ func updatePodRoleLabel(cli client.Client,
delete(pod.Labels, roleLabelKey)
delete(pod.Labels, rsmAccessModeLabelKey)
}
+
+ if pod.Annotations == nil {
+ pod.Annotations = map[string]string{}
+ }
+ pod.Annotations[constant.LastRoleSnapshotVersionAnnotationKey] = version
return cli.Patch(ctx, pod, patch)
}
@@ -170,9 +174,6 @@ func setMembersStatus(rsm *workloads.ReplicatedStateMachine, pods []corev1.Pod)
newMembersStatus := make([]workloads.MemberStatus, 0)
roleMap := composeRoleMap(*rsm)
for _, pod := range pods {
- if intctrlutil.GetPodRevision(&pod) != rsm.Status.UpdateRevision {
- continue
- }
if !intctrlutil.PodIsReadyWithLabel(pod) {
continue
}
@@ -189,11 +190,12 @@ func setMembersStatus(rsm *workloads.ReplicatedStateMachine, pods []corev1.Pod)
}
// sort and set
- rolePriorityMap := composeRolePriorityMap(*rsm)
+ rolePriorityMap := ComposeRolePriorityMap(rsm.Spec.Roles)
sortMembersStatus(newMembersStatus, rolePriorityMap)
rsm.Status.MembersStatus = newMembersStatus
}
+// getRoleName gets role name of pod 'pod'
func getRoleName(pod corev1.Pod) string {
return strings.ToLower(pod.Labels[constant.RoleLabelKey])
}
@@ -241,7 +243,7 @@ func getHeadlessSvcName(rsm workloads.ReplicatedStateMachine) string {
}
func findSvcPort(rsm workloads.ReplicatedStateMachine) int {
- if rsm.Spec.Service == nil {
+ if rsm.Spec.Service == nil || len(rsm.Spec.Service.Spec.Ports) == 0 {
return 0
}
port := rsm.Spec.Service.Spec.Ports[0]
@@ -524,14 +526,16 @@ func getLabels(rsm *workloads.ReplicatedStateMachine) map[string]string {
}
func getSvcSelector(rsm *workloads.ReplicatedStateMachine, headless bool) map[string]string {
- var leader *workloads.ReplicaRole
- for _, role := range rsm.Spec.Roles {
- if role.IsLeader && len(role.Name) > 0 {
- leader = &role
- break
+ selectors := make(map[string]string, 0)
+
+ if !headless {
+ for _, role := range rsm.Spec.Roles {
+ if role.IsLeader && len(role.Name) > 0 {
+ selectors[constant.RoleLabelKey] = role.Name
+ break
+ }
}
}
- selectors := make(map[string]string, 0)
if viper.GetBool(FeatureGateRSMCompatibilityMode) {
keys := []string{
@@ -544,18 +548,12 @@ func getSvcSelector(rsm *workloads.ReplicatedStateMachine, headless bool) map[st
selectors[key] = value
}
}
- if leader != nil && !headless {
- selectors[constant.RoleLabelKey] = leader.Name
- }
return selectors
}
for k, v := range rsm.Spec.Selector.MatchLabels {
selectors[k] = v
}
- if leader != nil && !headless {
- selectors[rsmAccessModeLabelKey] = string(leader.AccessMode)
- }
return selectors
}
diff --git a/internal/controller/rsm/utils_test.go b/internal/controller/rsm/utils_test.go
index 00a8959f523..814e73b960d 100644
--- a/internal/controller/rsm/utils_test.go
+++ b/internal/controller/rsm/utils_test.go
@@ -44,10 +44,10 @@ var _ = Describe("utils test", func() {
SetService(&corev1.Service{}).
SetRoles(roles).
GetObject()
- priorityMap = composeRolePriorityMap(*rsm)
+ priorityMap = ComposeRolePriorityMap(rsm.Spec.Roles)
})
- Context("composeRolePriorityMap function", func() {
+ Context("ComposeRolePriorityMap function", func() {
It("should work well", func() {
priorityList := []int{
leaderPriority,
@@ -63,7 +63,7 @@ var _ = Describe("utils test", func() {
})
})
- Context("sortPods function", func() {
+ Context("SortPods function", func() {
It("should work well", func() {
pods := []corev1.Pod{
*builder.NewPodBuilder(namespace, "pod-0").AddLabels(roleLabelKey, "follower").GetObject(),
@@ -76,7 +76,7 @@ var _ = Describe("utils test", func() {
}
expectedOrder := []string{"pod-4", "pod-2", "pod-3", "pod-6", "pod-1", "pod-0", "pod-5"}
- sortPods(pods, priorityMap, false)
+ SortPods(pods, priorityMap, false)
for i, pod := range pods {
Expect(pod.Name).Should(Equal(expectedOrder[i]))
}
diff --git a/internal/controllerutil/config_util.go b/internal/controllerutil/config_util.go
index c524365a3ac..0052dd066d5 100644
--- a/internal/controllerutil/config_util.go
+++ b/internal/controllerutil/config_util.go
@@ -21,6 +21,8 @@ package controllerutil
import (
"context"
+ "encoding/json"
+ "reflect"
"github.com/StudioSol/set"
appsv1 "k8s.io/api/apps/v1"
@@ -32,6 +34,7 @@ import (
"github.com/apecloud/kubeblocks/internal/configuration/core"
"github.com/apecloud/kubeblocks/internal/configuration/util"
"github.com/apecloud/kubeblocks/internal/configuration/validate"
+ "github.com/apecloud/kubeblocks/internal/constant"
)
type ConfigEventContext struct {
@@ -116,3 +119,50 @@ func fromUpdatedConfig(m map[string]string, sets *set.LinkedHashSetString) map[s
}
return r
}
+
+// IsApplyConfigChanged checks if the configuration is changed
+func IsApplyConfigChanged(configMap *corev1.ConfigMap, item v1alpha1.ConfigurationItemDetail) bool {
+ if configMap == nil {
+ return false
+ }
+
+ lastAppliedVersion, ok := configMap.Annotations[constant.ConfigAppliedVersionAnnotationKey]
+ if !ok {
+ return false
+ }
+ var target v1alpha1.ConfigurationItemDetail
+ if err := json.Unmarshal([]byte(lastAppliedVersion), &target); err != nil {
+ return false
+ }
+
+ return reflect.DeepEqual(target, item)
+}
+
+// IsRerender checks if the configuration template is changed
+func IsRerender(configMap *corev1.ConfigMap, item v1alpha1.ConfigurationItemDetail) bool {
+ if configMap == nil {
+ return true
+ }
+ if item.Version == "" {
+ return false
+ }
+
+ version, ok := configMap.Annotations[constant.CMConfigurationTemplateVersion]
+ if !ok || version != item.Version {
+ return true
+ }
+ return false
+}
+
+// GetConfigSpecReconcilePhase gets the configuration phase
+func GetConfigSpecReconcilePhase(configMap *corev1.ConfigMap,
+ item v1alpha1.ConfigurationItemDetail,
+ status *v1alpha1.ConfigurationItemDetailStatus) v1alpha1.ConfigurationPhase {
+ if status == nil || status.Phase == "" {
+ return v1alpha1.CCreatingPhase
+ }
+ if !IsApplyConfigChanged(configMap, item) {
+ return v1alpha1.CPendingPhase
+ }
+ return status.Phase
+}
diff --git a/internal/controllerutil/config_util_test.go b/internal/controllerutil/config_util_test.go
index 23edc80ba64..7d91f33540d 100644
--- a/internal/controllerutil/config_util_test.go
+++ b/internal/controllerutil/config_util_test.go
@@ -28,10 +28,13 @@ import (
. "github.com/onsi/gomega"
"github.com/StudioSol/set"
+ corev1 "k8s.io/api/core/v1"
"github.com/apecloud/kubeblocks/apis/apps/v1alpha1"
"github.com/apecloud/kubeblocks/internal/configuration/core"
cfgutil "github.com/apecloud/kubeblocks/internal/configuration/util"
+ "github.com/apecloud/kubeblocks/internal/constant"
+ "github.com/apecloud/kubeblocks/internal/controller/builder"
testapps "github.com/apecloud/kubeblocks/internal/testutil/apps"
testutil "github.com/apecloud/kubeblocks/internal/testutil/k8s"
"github.com/apecloud/kubeblocks/test/testdata"
@@ -81,6 +84,134 @@ func TestFromUpdatedConfig(t *testing.T) {
}
}
+func TestIsRerender(t *testing.T) {
+ type args struct {
+ cm *corev1.ConfigMap
+ item v1alpha1.ConfigurationItemDetail
+ }
+ tests := []struct {
+ name string
+ args args
+ want bool
+ }{{
+
+ name: "test",
+ args: args{
+ cm: nil,
+ item: v1alpha1.ConfigurationItemDetail{
+ Name: "test",
+ },
+ },
+ want: true,
+ }, {
+ name: "test",
+ args: args{
+ cm: builder.NewConfigMapBuilder("default", "test").GetObject(),
+ item: v1alpha1.ConfigurationItemDetail{
+ Name: "test",
+ },
+ },
+ want: false,
+ }, {
+ name: "test",
+ args: args{
+ cm: builder.NewConfigMapBuilder("default", "test").
+ GetObject(),
+ item: v1alpha1.ConfigurationItemDetail{
+ Name: "test",
+ Version: "v1",
+ },
+ },
+ want: true,
+ }, {
+ name: "test",
+ args: args{
+ cm: builder.NewConfigMapBuilder("default", "test").
+ AddAnnotations(constant.CMConfigurationTemplateVersion, "v1").
+ GetObject(),
+ item: v1alpha1.ConfigurationItemDetail{
+ Name: "test",
+ Version: "v2",
+ },
+ },
+ want: true,
+ }, {
+ name: "test",
+ args: args{
+ cm: builder.NewConfigMapBuilder("default", "test").
+ AddAnnotations(constant.CMConfigurationTemplateVersion, "v1").
+ GetObject(),
+ item: v1alpha1.ConfigurationItemDetail{
+ Name: "test",
+ Version: "v1",
+ },
+ },
+ want: false,
+ }}
+ for _, tt := range tests {
+ t.Run(tt.name, func(t *testing.T) {
+ if got := IsRerender(tt.args.cm, tt.args.item); got != tt.want {
+ t.Errorf("IsRerender() = %v, want %v", got, tt.want)
+ }
+ })
+ }
+}
+
+func TestGetConfigSpecReconcilePhase(t *testing.T) {
+ type args struct {
+ cm *corev1.ConfigMap
+ item v1alpha1.ConfigurationItemDetail
+ status *v1alpha1.ConfigurationItemDetailStatus
+ }
+ tests := []struct {
+ name string
+ args args
+ want v1alpha1.ConfigurationPhase
+ }{{
+ name: "test",
+ args: args{
+ cm: nil,
+ item: v1alpha1.ConfigurationItemDetail{
+ Name: "test",
+ },
+ },
+ want: v1alpha1.CCreatingPhase,
+ }, {
+ name: "test",
+ args: args{
+ cm: builder.NewConfigMapBuilder("default", "test").GetObject(),
+ item: v1alpha1.ConfigurationItemDetail{
+ Name: "test",
+ },
+ status: &v1alpha1.ConfigurationItemDetailStatus{
+ Phase: v1alpha1.CInitPhase,
+ },
+ },
+ want: v1alpha1.CPendingPhase,
+ }, {
+ name: "test",
+ args: args{
+ cm: builder.NewConfigMapBuilder("default", "test").
+ AddAnnotations(constant.ConfigAppliedVersionAnnotationKey, `{"name":"test"}`).
+ GetObject(),
+ item: v1alpha1.ConfigurationItemDetail{
+ Name: "test",
+ },
+ status: &v1alpha1.ConfigurationItemDetailStatus{
+ Phase: v1alpha1.CUpgradingPhase,
+ },
+ },
+ want: v1alpha1.CUpgradingPhase,
+ }}
+ for _, tt := range tests {
+ t.Run(tt.name, func(t *testing.T) {
+ if got := GetConfigSpecReconcilePhase(tt.args.cm, tt.args.item, tt.args.status); got != tt.want {
+ t.Errorf("GetConfigSpecReconcilePhase() = %v, want %v", got, tt.want)
+ }
+ })
+ }
+}
+
var _ = Describe("config_util", func() {
var k8sMockClient *testutil.K8sClientMockHelper
diff --git a/internal/controllerutil/controller_common.go b/internal/controllerutil/controller_common.go
index 45dccb00fc7..1df1ab8c850 100644
--- a/internal/controllerutil/controller_common.go
+++ b/internal/controllerutil/controller_common.go
@@ -233,8 +233,8 @@ func BackgroundDeleteObject(cli client.Client, ctx context.Context, obj client.O
PropagationPolicy: &deletePropagation,
}
- if err := cli.Delete(ctx, obj, deleteOptions); err != nil && !apierrors.IsNotFound(err) {
- return err
+ if err := cli.Delete(ctx, obj, deleteOptions); err != nil {
+ return client.IgnoreNotFound(err)
}
return nil
}
diff --git a/internal/controllerutil/errors.go b/internal/controllerutil/errors.go
index 9cc552195b1..e56e54f071c 100644
--- a/internal/controllerutil/errors.go
+++ b/internal/controllerutil/errors.go
@@ -22,7 +22,6 @@ package controllerutil
import (
"errors"
"fmt"
- "strings"
)
type Error struct {
@@ -48,21 +47,12 @@ const (
ErrorTypeRequeue ErrorType = "Requeue" // requeue for reconcile.
- // ErrorType for backup
- ErrorTypeBackupNotSupported ErrorType = "BackupNotSupported" // this backup type not supported
- ErrorTypeBackupPVTemplateNotFound ErrorType = "BackupPVTemplateNotFound" // this pv template not found
- ErrorTypeBackupNotCompleted ErrorType = "BackupNotCompleted" // report backup not completed.
- ErrorTypeBackupPVCNameIsEmpty ErrorType = "BackupPVCNameIsEmpty" // pvc name for backup is empty
- ErrorTypeBackupJobFailed ErrorType = "BackupJobFailed" // backup job failed
- ErrorTypeStorageNotMatch ErrorType = "ErrorTypeStorageNotMatch"
- ErrorTypeReconfigureFailed ErrorType = "ErrorTypeReconfigureFailed"
- ErrorTypeInvalidLogfileBackupName ErrorType = "InvalidLogfileBackupName"
- ErrorTypeBackupScheduleDisabled ErrorType = "BackupScheduleDisabled"
- ErrorTypeLogfileScheduleDisabled ErrorType = "LogfileScheduleDisabled"
+ ErrorTypeFatal ErrorType = "Fatal" // fatal error
// ErrorType for cluster controller
- ErrorTypeBackupFailed ErrorType = "BackupFailed"
- ErrorTypeNeedWaiting ErrorType = "NeedWaiting" // waiting for next reconcile
+ ErrorTypeBackupFailed ErrorType = "BackupFailed"
+ ErrorTypeRestoreFailed ErrorType = "RestoreFailed"
+ ErrorTypeNeedWaiting ErrorType = "NeedWaiting" // waiting for next reconcile
// ErrorType for preflight
ErrorTypePreflightCommon = "PreflightCommon"
@@ -114,37 +104,7 @@ func IsNotFound(err error) bool {
return IsTargetError(err, ErrorTypeNotFound)
}
-// NewBackupNotSupported returns a new Error with ErrorTypeBackupNotSupported.
-func NewBackupNotSupported(backupType, backupPolicyName string) *Error {
- return NewErrorf(ErrorTypeBackupNotSupported, `backup type "%s" not supported by backup policy "%s"`, backupType, backupPolicyName)
-}
-
-// NewBackupPVTemplateNotFound returns a new Error with ErrorTypeBackupPVTemplateNotFound.
-func NewBackupPVTemplateNotFound(cmName, cmNamespace string) *Error {
- return NewErrorf(ErrorTypeBackupPVTemplateNotFound, `"the persistentVolume template is empty in the configMap %s/%s", pvConfig.Namespace, pvConfig.Name`, cmNamespace, cmName)
-}
-
-// NewBackupPVCNameIsEmpty returns a new Error with ErrorTypeBackupPVCNameIsEmpty.
-func NewBackupPVCNameIsEmpty(backupType, backupPolicyName string) *Error {
- return NewErrorf(ErrorTypeBackupPVCNameIsEmpty, `the persistentVolumeClaim name of spec.%s is empty in BackupPolicy "%s"`, strings.ToLower(backupType), backupPolicyName)
-}
-
-// NewBackupJobFailed returns a new Error with ErrorTypeBackupJobFailed.
-func NewBackupJobFailed(jobName string) *Error {
- return NewErrorf(ErrorTypeBackupJobFailed, `backup job "%s" failed`, jobName)
-}
-
-// NewInvalidLogfileBackupName returns a new Error with ErrorTypeInvalidLogfileBackupName.
-func NewInvalidLogfileBackupName(backupPolicyName string) *Error {
- return NewErrorf(ErrorTypeInvalidLogfileBackupName, `backup name is incorrect for logfile, you can create the logfile backup by enabling the schedule in BackupPolicy "%s"`, backupPolicyName)
-}
-
-// NewBackupScheduleDisabled returns a new Error with ErrorTypeBackupScheduleDisabled.
-func NewBackupScheduleDisabled(backupType, backupPolicyName string) *Error {
- return NewErrorf(ErrorTypeBackupScheduleDisabled, `%s schedule is disabled, you can enable spec.schedule.%s in BackupPolicy "%s"`, backupType, backupType, backupPolicyName)
-}
-
-// NewBackupLogfileScheduleDisabled returns a new Error with ErrorTypeLogfileScheduleDisabled.
-func NewBackupLogfileScheduleDisabled(backupToolName string) *Error {
- return NewErrorf(ErrorTypeLogfileScheduleDisabled, `BackupTool "%s" of the backup relies on logfile. Please enable the logfile scheduling firstly`, backupToolName)
+// NewFatalError returns a new Error with ErrorTypeFatal
+func NewFatalError(message string) *Error {
+ return NewErrorf(ErrorTypeFatal, message)
}
diff --git a/internal/controllerutil/errors_test.go b/internal/controllerutil/errors_test.go
index ff9ce39fe81..469df4b7b5b 100644
--- a/internal/controllerutil/errors_test.go
+++ b/internal/controllerutil/errors_test.go
@@ -20,67 +20,11 @@ along with this program. If not, see .
package controllerutil
import (
- "fmt"
"testing"
"github.com/pkg/errors"
)
-func TestNerError(t *testing.T) {
- err1 := NewError(ErrorTypeBackupNotCompleted, "test c2")
- if err1.Error() != "test c2" {
- t.Error("NewErrorf failed")
- }
-}
-
-func TestNerErrorf(t *testing.T) {
- err1 := NewErrorf(ErrorTypeBackupNotCompleted, "test %s %s", "c1", "c2")
- if err1.Error() != "test c1 c2" {
- t.Error("NewErrorf failed")
- }
- testError := fmt.Errorf("test: %w", err1)
- if !errors.Is(testError, err1) {
- t.Error("errors.Is failed")
- }
-
- var target *Error
- if !errors.As(testError, &target) {
- t.Error("errors.As failed")
- }
-}
-
-func TestNewErrors(t *testing.T) {
- backupNotSupported := NewBackupNotSupported("datafile", "policy-test")
- if !IsTargetError(backupNotSupported, ErrorTypeBackupNotSupported) {
- t.Error("should be error of BackupNotSupported")
- }
- pvTemplateNotFound := NewBackupPVTemplateNotFound("configName", "default")
- if !IsTargetError(pvTemplateNotFound, ErrorTypeBackupPVTemplateNotFound) {
- t.Error("should be error of BackupPVTemplateNotFound")
- }
- pvsIsEmpty := NewBackupPVCNameIsEmpty("datafile", "policy-test1")
- if !IsTargetError(pvsIsEmpty, ErrorTypeBackupPVCNameIsEmpty) {
- t.Error("should be error of BackupPVCNameIsEmpty")
- }
- jobFailed := NewBackupJobFailed("jobName")
- if !IsTargetError(jobFailed, ErrorTypeBackupJobFailed) {
- t.Error("should be error of BackupJobFailed")
- }
-}
-
-func TestUnwrapControllerError(t *testing.T) {
- backupNotSupported := NewBackupNotSupported("datafile", "policy-test")
- newErr := UnwrapControllerError(backupNotSupported)
- if newErr == nil {
- t.Error("should unwrap a controller error, but got nil")
- }
- err := errors.New("test error")
- newErr = UnwrapControllerError(err)
- if newErr != nil {
- t.Errorf("should not unwrap a controller error, but got: %v", newErr)
- }
-}
-
func TestIsTargetError(t *testing.T) {
var err1 error
if IsTargetError(err1, ErrorWaitCacheRefresh) {
diff --git a/internal/controllerutil/pod_utils.go b/internal/controllerutil/pod_utils.go
index 4e81d0086ce..68d79701789 100644
--- a/internal/controllerutil/pod_utils.go
+++ b/internal/controllerutil/pod_utils.go
@@ -34,6 +34,7 @@ import (
appsv1alpha1 "github.com/apecloud/kubeblocks/apis/apps/v1alpha1"
"github.com/apecloud/kubeblocks/internal/constant"
+ viper "github.com/apecloud/kubeblocks/internal/viperx"
)
// statefulPodRegex is a regular expression that extracts the parent StatefulSet and ordinal from the Name of a Pod
@@ -355,14 +356,30 @@ func GetProbeHTTPPort(pod *corev1.Pod) (int32, error) {
return GetPortByPortName(pod, constant.ProbeHTTPPortName)
}
+// GuessLorryHTTPPort guesses lorry container and serving port.
+// TODO(xuriwuyun): should provide a deterministic way to find the lorry serving port.
+func GuessLorryHTTPPort(pod *corev1.Pod) (int32, error) {
+ lorryImage := viper.GetString(constant.KBToolsImage)
+ for _, container := range pod.Spec.Containers {
+ if container.Image != lorryImage {
+ continue
+ }
+ if len(container.Ports) > 0 {
+ return container.Ports[0].ContainerPort, nil
+ }
+ }
+ return 0, fmt.Errorf("lorry port not found")
+}
+
// GetProbeContainerName gets the probe container from pod
func GetProbeContainerName(pod *corev1.Pod) (string, error) {
+ lorryImage := viper.GetString(constant.KBToolsImage)
for _, container := range pod.Spec.Containers {
- if container.Name == constant.RoleProbeContainerName {
- return constant.RoleProbeContainerName, nil
+ if container.Image == lorryImage {
+ return container.Name, nil
}
}
- return "", fmt.Errorf("container %s not found", constant.RoleProbeContainerName)
+ return "", fmt.Errorf("container %s not found", lorryImage)
}
@@ -411,14 +428,18 @@ func (c ByPodName) Less(i, j int) bool {
// BuildPodHostDNS builds the host dns of pod.
// ref: https://kubernetes.io/docs/concepts/services-networking/dns-pod-service/
func BuildPodHostDNS(pod *corev1.Pod) string {
+ if pod == nil {
+ return ""
+ }
// build pod dns string
// ref: https://kubernetes.io/docs/concepts/services-networking/dns-pod-service/
- hostDNS := []string{pod.Name}
- if pod.Spec.Hostname != "" {
- hostDNS[0] = pod.Spec.Hostname
- }
if pod.Spec.Subdomain != "" {
+ hostDNS := []string{pod.Name}
+ if pod.Spec.Hostname != "" {
+ hostDNS[0] = pod.Spec.Hostname
+ }
hostDNS = append(hostDNS, pod.Spec.Subdomain)
+ return strings.Join(hostDNS, ".")
}
- return strings.Join(hostDNS, ".")
+ return pod.Status.PodIP
}
diff --git a/internal/controllerutil/util.go b/internal/controllerutil/util.go
index e3249ca7577..3d9804e60b8 100644
--- a/internal/controllerutil/util.go
+++ b/internal/controllerutil/util.go
@@ -21,6 +21,7 @@ package controllerutil
import (
"context"
+ "reflect"
corev1 "k8s.io/api/core/v1"
"k8s.io/apimachinery/pkg/runtime"
@@ -94,3 +95,14 @@ func IsRSMEnabled() bool {
}
return true
}
+
+func IsNil(i interface{}) bool {
+ if i == nil {
+ return true
+ }
+ switch reflect.TypeOf(i).Kind() {
+ case reflect.Ptr, reflect.Map, reflect.Array, reflect.Chan, reflect.Slice:
+ return reflect.ValueOf(i).IsNil()
+ }
+ return false
+}
diff --git a/controllers/apps/components/rsm_workload.go b/internal/dataprotection/action/action.go
similarity index 55%
rename from controllers/apps/components/rsm_workload.go
rename to internal/dataprotection/action/action.go
index 72039d9fcb6..d3e51216b31 100644
--- a/controllers/apps/components/rsm_workload.go
+++ b/internal/dataprotection/action/action.go
@@ -17,30 +17,35 @@ You should have received a copy of the GNU Affero General Public License
along with this program. If not, see .
*/
-package components
+package action
import (
- "github.com/apecloud/kubeblocks/internal/controller/factory"
+ "context"
+
+ "k8s.io/apimachinery/pkg/runtime"
+ "k8s.io/client-go/rest"
+ "k8s.io/client-go/tools/record"
"sigs.k8s.io/controller-runtime/pkg/client"
+
+ dpv1alpha1 "github.com/apecloud/kubeblocks/apis/dataprotection/v1alpha1"
)
-type rsmComponentWorkloadBuilder struct {
- componentWorkloadBuilderBase
-}
+type Action interface {
+ // Execute executes the action.
+ Execute(ctx Context) (*dpv1alpha1.ActionStatus, error)
-var _ componentWorkloadBuilder = &rsmComponentWorkloadBuilder{}
+ // GetName returns the Name of the action.
+ GetName() string
-func (b *rsmComponentWorkloadBuilder) BuildWorkload() componentWorkloadBuilder {
- buildfn := func() ([]client.Object, error) {
- component := b.Comp.GetSynthesizedComponent()
- obj, err := factory.BuildRSM(b.ReqCtx, b.Comp.GetCluster(), component, b.EnvConfig.Name)
- if err != nil {
- return nil, err
- }
+ // Type returns the type of the action.
+ Type() dpv1alpha1.ActionType
+}
- b.Workload = obj
+type Context struct {
+ Ctx context.Context
+ Client client.Client
+ Recorder record.EventRecorder
- return nil, nil // don't return sts here
- }
- return b.BuildWrapper(buildfn)
+ Scheme *runtime.Scheme
+ RestClientConfig *rest.Config
}
diff --git a/internal/dataprotection/action/action_create_vs.go b/internal/dataprotection/action/action_create_vs.go
new file mode 100644
index 00000000000..43ccfcc9e94
--- /dev/null
+++ b/internal/dataprotection/action/action_create_vs.go
@@ -0,0 +1,261 @@
+/*
+Copyright (C) 2022-2023 ApeCloud Co., Ltd
+
+This file is part of KubeBlocks project
+
+This program is free software: you can redistribute it and/or modify
+it under the terms of the GNU Affero General Public License as published by
+the Free Software Foundation, either version 3 of the License, or
+(at your option) any later version.
+
+This program is distributed in the hope that it will be useful
+but WITHOUT ANY WARRANTY; without even the implied warranty of
+MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+GNU Affero General Public License for more details.
+
+You should have received a copy of the GNU Affero General Public License
+along with this program. If not, see .
+*/
+
+package action
+
+import (
+ "context"
+ "fmt"
+ "strings"
+
+ vsv1 "github.com/kubernetes-csi/external-snapshotter/client/v6/apis/volumesnapshot/v1"
+ "github.com/pkg/errors"
+ corev1 "k8s.io/api/core/v1"
+ storagev1 "k8s.io/api/storage/v1"
+ metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
+ "sigs.k8s.io/controller-runtime/pkg/client"
+ "sigs.k8s.io/controller-runtime/pkg/controller/controllerutil"
+
+ dpv1alpha1 "github.com/apecloud/kubeblocks/apis/dataprotection/v1alpha1"
+ intctrlutil "github.com/apecloud/kubeblocks/internal/controllerutil"
+ "github.com/apecloud/kubeblocks/internal/dataprotection/builder"
+ dptypes "github.com/apecloud/kubeblocks/internal/dataprotection/types"
+ dputils "github.com/apecloud/kubeblocks/internal/dataprotection/utils"
+)
+
+// CreateVolumeSnapshotAction is an action that creates the volume snapshot.
+type CreateVolumeSnapshotAction struct {
+ // Name is the Name of the action.
+ Name string
+
+ // Owner is the owner of the volume snapshot.
+ Owner client.Object
+
+ // ObjectMeta is the metadata of the volume snapshot.
+ ObjectMeta metav1.ObjectMeta
+
+ // PersistentVolumeClaimWrappers is the list of persistent volume claims wrapper to snapshot.
+ PersistentVolumeClaimWrappers []PersistentVolumeClaimWrapper
+}
+
+type PersistentVolumeClaimWrapper struct {
+ VolumeName string
+ PersistentVolumeClaim corev1.PersistentVolumeClaim
+}
+
+func NewPersistentVolumeClaimWrapper(pvc corev1.PersistentVolumeClaim, volumeName string) PersistentVolumeClaimWrapper {
+ return PersistentVolumeClaimWrapper{PersistentVolumeClaim: pvc, VolumeName: volumeName}
+}
+
+var configVolumeSnapshotError = []string{
+ "Failed to set default snapshot class with error",
+ "Failed to get snapshot class with error",
+ "Failed to create snapshot content with error cannot find CSI PersistentVolumeSource for volume",
+}
+
+func (c *CreateVolumeSnapshotAction) GetName() string {
+ return c.Name
+}
+
+func (c *CreateVolumeSnapshotAction) Type() dpv1alpha1.ActionType {
+ return dpv1alpha1.ActionTypeNone
+}
+
+func (c *CreateVolumeSnapshotAction) Execute(ctx Context) (*dpv1alpha1.ActionStatus, error) {
+ sb := newStatusBuilder(c)
+ handleErr := func(err error) (*dpv1alpha1.ActionStatus, error) {
+ return sb.withErr(err).build(), err
+ }
+
+ if err := c.validate(); err != nil {
+ return handleErr(err)
+ }
+
+ vsCli := intctrlutil.VolumeSnapshotCompatClient{
+ Client: ctx.Client,
+ Ctx: ctx.Ctx,
+ }
+
+ var (
+ ok bool
+ err error
+ snap *vsv1.VolumeSnapshot
+ )
+ for _, w := range c.PersistentVolumeClaimWrappers {
+ key := client.ObjectKey{
+ Namespace: w.PersistentVolumeClaim.Namespace,
+ Name: dputils.GetBackupVolumeSnapshotName(c.ObjectMeta.Name, w.VolumeName),
+ }
+ // create volume snapshot
+ if err = c.createVolumeSnapshotIfNotExist(ctx, vsCli, &w.PersistentVolumeClaim, key); err != nil {
+ return handleErr(err)
+ }
+
+ ok, snap, err = ensureVolumeSnapshotReady(vsCli, key)
+ if err != nil {
+ return handleErr(err)
+ }
+
+ if !ok {
+ return sb.startTimestamp(&snap.CreationTimestamp).build(), nil
+ }
+ }
+
+ // volume snapshot is ready and status is not error
+ // TODO(ldm): now only support one volume to take snapshot, set its time, size to status
+ return sb.phase(dpv1alpha1.ActionPhaseCompleted).
+ phase(dpv1alpha1.ActionPhaseCompleted).
+ totalSize(snap.Status.RestoreSize.String()).
+ timeRange(snap.Status.CreationTime, snap.Status.CreationTime).
+ build(), nil
+}
+
+func (c *CreateVolumeSnapshotAction) validate() error {
+ if len(c.PersistentVolumeClaimWrappers) == 0 {
+ return errors.New("persistent volume claims are required")
+ }
+ if len(c.PersistentVolumeClaimWrappers) > 1 {
+ return errors.New("only one persistent volume claim is supported")
+ }
+ return nil
+}
+
+// createVolumeSnapshotIfNotExist check volume snapshot exists, if not, create it.
+func (c *CreateVolumeSnapshotAction) createVolumeSnapshotIfNotExist(ctx Context,
+ vsCli intctrlutil.VolumeSnapshotCompatClient,
+ pvc *corev1.PersistentVolumeClaim,
+ key client.ObjectKey) error {
+ var (
+ err error
+ vsc *vsv1.VolumeSnapshotClass
+ )
+
+ snap := &vsv1.VolumeSnapshot{}
+ exists, err := vsCli.CheckResourceExists(key, snap)
+ if err != nil {
+ return err
+ }
+
+ // if the volume snapshot already exists, skip creating it.
+ if exists {
+ return nil
+ }
+
+ // create volume snapshot
+ if pvc.Spec.StorageClassName != nil && *pvc.Spec.StorageClassName != "" {
+ vsc, err = createVolumeSnapshotClassIfNotExist(ctx.Ctx, ctx.Client, vsCli, *pvc.Spec.StorageClassName)
+ if err != nil {
+ return err
+ }
+ }
+
+ c.ObjectMeta.Name = key.Name
+ c.ObjectMeta.Namespace = key.Namespace
+
+ // create volume snapshot
+ snap = &vsv1.VolumeSnapshot{
+ ObjectMeta: c.ObjectMeta,
+ Spec: vsv1.VolumeSnapshotSpec{
+ Source: vsv1.VolumeSnapshotSource{
+ PersistentVolumeClaimName: &pvc.Name,
+ },
+ },
+ }
+
+ if vsc != nil {
+ snap.Spec.VolumeSnapshotClassName = &vsc.Name
+ }
+
+ controllerutil.AddFinalizer(snap, dptypes.DataProtectionFinalizerName)
+ if err = setControllerReference(c.Owner, snap, ctx.Scheme); err != nil {
+ return err
+ }
+
+ msg := fmt.Sprintf("creating volume snapshot %s/%s", snap.Namespace, snap.Name)
+ ctx.Recorder.Event(c.Owner, corev1.EventTypeNormal, "CreatingVolumeSnapshot", msg)
+ if err = ctx.Client.Create(ctx.Ctx, snap); err != nil {
+ return err
+ }
+ return nil
+}
+
+func createVolumeSnapshotClassIfNotExist(
+ ctx context.Context,
+ cli client.Client,
+ vsCli intctrlutil.VolumeSnapshotCompatClient,
+ scName string) (*vsv1.VolumeSnapshotClass, error) {
+ scObj := storagev1.StorageClass{}
+ // ignore if not found storage class, use the default volume snapshot class
+ if err := cli.Get(ctx, client.ObjectKey{Name: scName}, &scObj); client.IgnoreNotFound(err) != nil {
+ return nil, err
+ }
+
+ vscList := vsv1.VolumeSnapshotClassList{}
+ if err := vsCli.List(&vscList); err != nil {
+ return nil, err
+ }
+ for _, item := range vscList.Items {
+ if item.Driver == scObj.Provisioner {
+ return item.DeepCopy(), nil
+ }
+ }
+
+ // not found matched volume snapshot class, create one
+ vscName := fmt.Sprintf("vsc-%s-%s", scName, scObj.UID[:8])
+ newVsc := builder.BuildVolumeSnapshotClass(vscName, scObj.Provisioner)
+ if err := vsCli.Create(newVsc); err != nil {
+ return nil, err
+ }
+ return newVsc, nil
+}
+
+func ensureVolumeSnapshotReady(
+ vsCli intctrlutil.VolumeSnapshotCompatClient,
+ key client.ObjectKey) (bool, *vsv1.VolumeSnapshot, error) {
+ snap := &vsv1.VolumeSnapshot{}
+ // not found, continue the creation process
+ exists, err := vsCli.CheckResourceExists(key, snap)
+ if err != nil {
+ return false, nil, err
+ }
+ if exists && snap.Status != nil {
+ // check if snapshot status throws an error, e.g. csi does not support volume snapshot
+ if isVolumeSnapshotConfigError(snap) {
+ return false, nil, errors.New(*snap.Status.Error.Message)
+ }
+ if snap.Status.ReadyToUse != nil && *snap.Status.ReadyToUse {
+ return true, snap, nil
+ }
+ }
+ return false, snap, nil
+}
+
+func isVolumeSnapshotConfigError(snap *vsv1.VolumeSnapshot) bool {
+ if snap.Status == nil || snap.Status.Error == nil || snap.Status.Error.Message == nil {
+ return false
+ }
+ for _, errMsg := range configVolumeSnapshotError {
+ if strings.Contains(*snap.Status.Error.Message, errMsg) {
+ return true
+ }
+ }
+ return false
+}
+
+var _ Action = &CreateVolumeSnapshotAction{}
diff --git a/internal/dataprotection/action/action_create_vs_test.go b/internal/dataprotection/action/action_create_vs_test.go
new file mode 100644
index 00000000000..4258ec3fe48
--- /dev/null
+++ b/internal/dataprotection/action/action_create_vs_test.go
@@ -0,0 +1,102 @@
+/*
+Copyright (C) 2022-2023 ApeCloud Co., Ltd
+
+This file is part of KubeBlocks project
+
+This program is free software: you can redistribute it and/or modify
+it under the terms of the GNU Affero General Public License as published by
+the Free Software Foundation, either version 3 of the License, or
+(at your option) any later version.
+
+This program is distributed in the hope that it will be useful
+but WITHOUT ANY WARRANTY; without even the implied warranty of
+MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+GNU Affero General Public License for more details.
+
+You should have received a copy of the GNU Affero General Public License
+along with this program. If not, see .
+*/
+
+package action_test
+
+import (
+ . "github.com/onsi/ginkgo/v2"
+ . "github.com/onsi/gomega"
+
+ vsv1 "github.com/kubernetes-csi/external-snapshotter/client/v6/apis/volumesnapshot/v1"
+ metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
+ "sigs.k8s.io/controller-runtime/pkg/client"
+
+ dpv1alpha1 "github.com/apecloud/kubeblocks/apis/dataprotection/v1alpha1"
+ "github.com/apecloud/kubeblocks/internal/constant"
+ "github.com/apecloud/kubeblocks/internal/dataprotection/action"
+ dputils "github.com/apecloud/kubeblocks/internal/dataprotection/utils"
+ "github.com/apecloud/kubeblocks/internal/generics"
+ testapps "github.com/apecloud/kubeblocks/internal/testutil/apps"
+ testdp "github.com/apecloud/kubeblocks/internal/testutil/dataprotection"
+ viper "github.com/apecloud/kubeblocks/internal/viperx"
+)
+
+var _ = Describe("CreateVolumeSnapshotAction Test", func() {
+ const (
+ actionName = "test-create-vs-action"
+ pvcName = "test-pvc"
+ volumeName = "test-volume"
+ )
+
+ cleanEnv := func() {
+ By("clean resources")
+ inNS := client.InNamespace(testCtx.DefaultNamespace)
+ testapps.ClearResourcesWithRemoveFinalizerOption(&testCtx, generics.BackupSignature, true, inNS)
+ testapps.ClearResourcesWithRemoveFinalizerOption(&testCtx, generics.PersistentVolumeClaimSignature, true, inNS)
+ testapps.ClearResourcesWithRemoveFinalizerOption(&testCtx, generics.VolumeSnapshotSignature, true, inNS)
+ }
+
+ BeforeEach(func() {
+ cleanEnv()
+ viper.Set(constant.KBToolsImage, testdp.KBToolImage)
+ })
+
+ AfterEach(func() {
+ cleanEnv()
+ viper.Set(constant.KBToolsImage, "")
+ })
+
+ Context("create action that create volume snapshot", func() {
+ It("should return error when PVC is empty", func() {
+ act := &action.CreateVolumeSnapshotAction{}
+ status, err := act.Execute(buildActionCtx())
+ Expect(err).To(HaveOccurred())
+ Expect(status.Phase).Should(Equal(dpv1alpha1.ActionPhaseFailed))
+ })
+
+ It("should success to execute action", func() {
+ act := &action.CreateVolumeSnapshotAction{
+ Name: actionName,
+ Owner: testdp.NewFakeBackup(&testCtx, nil),
+ ObjectMeta: metav1.ObjectMeta{
+ Namespace: testCtx.DefaultNamespace,
+ Name: actionName,
+ },
+ PersistentVolumeClaimWrappers: []action.PersistentVolumeClaimWrapper{
+ {
+ PersistentVolumeClaim: *testdp.NewFakePVC(&testCtx, pvcName),
+ VolumeName: volumeName,
+ },
+ },
+ }
+
+ By("execute action, its status should be running")
+ status, err := act.Execute(buildActionCtx())
+ Expect(err).ShouldNot(HaveOccurred())
+ Expect(status.Phase).Should(Equal(dpv1alpha1.ActionPhaseRunning))
+
+ By("check volume snapshot be created")
+ key := client.ObjectKey{
+ Namespace: testCtx.DefaultNamespace,
+ Name: dputils.GetBackupVolumeSnapshotName(actionName, volumeName),
+ }
+ Eventually(testapps.CheckObjExists(&testCtx, key, &vsv1.VolumeSnapshot{}, true)).Should(Succeed())
+ })
+ })
+})
diff --git a/internal/dataprotection/action/action_exec.go b/internal/dataprotection/action/action_exec.go
new file mode 100644
index 00000000000..c004b7ad3f9
--- /dev/null
+++ b/internal/dataprotection/action/action_exec.go
@@ -0,0 +1,110 @@
+/*
+Copyright (C) 2022-2023 ApeCloud Co., Ltd
+
+This file is part of KubeBlocks project
+
+This program is free software: you can redistribute it and/or modify
+it under the terms of the GNU Affero General Public License as published by
+the Free Software Foundation, either version 3 of the License, or
+(at your option) any later version.
+
+This program is distributed in the hope that it will be useful
+but WITHOUT ANY WARRANTY; without even the implied warranty of
+MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+GNU Affero General Public License for more details.
+
+You should have received a copy of the GNU Affero General Public License
+along with this program. If not, see .
+*/
+
+package action
+
+import (
+ "github.com/pkg/errors"
+ corev1 "k8s.io/api/core/v1"
+ metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
+
+ dpv1alpha1 "github.com/apecloud/kubeblocks/apis/dataprotection/v1alpha1"
+ "github.com/apecloud/kubeblocks/internal/constant"
+ viper "github.com/apecloud/kubeblocks/internal/viperx"
+)
+
+// ExecAction is an action that executes a command on a pod.
+// This action will create a job to execute the command.
+type ExecAction struct {
+ JobAction
+
+ // PodName is the Name of the pod to execute the command on.
+ PodName string
+
+ // Namespace is the Namespace of the pod to execute the command on.
+ Namespace string
+
+ // Command is the command to execute.
+ Command []string
+
+ // Container is the container to execute the command on.
+ Container string
+
+ // ServiceAccountName is the service account to use to build the job object.
+ ServiceAccountName string
+
+ // Timeout is the timeout for the command.
+ Timeout metav1.Duration
+}
+
+func (e *ExecAction) Execute(ctx Context) (*dpv1alpha1.ActionStatus, error) {
+ if err := e.validate(); err != nil {
+ return nil, err
+ }
+ e.JobAction.PodSpec = e.buildPodSpec()
+ return e.JobAction.Execute(ctx)
+}
+
+func (e *ExecAction) validate() error {
+ if e.PodName == "" {
+ return errors.New("pod name is required")
+ }
+ if e.Namespace == "" {
+ return errors.New("namespace is required")
+ }
+ if len(e.Command) == 0 {
+ return errors.New("command is required")
+ }
+ return nil
+}
+
+func (e *ExecAction) buildPodSpec() *corev1.PodSpec {
+ return &corev1.PodSpec{
+ RestartPolicy: corev1.RestartPolicyNever,
+ ServiceAccountName: e.ServiceAccountName,
+ Containers: []corev1.Container{
+ {
+ Name: e.Name,
+ Image: viper.GetString(constant.KBToolsImage),
+ ImagePullPolicy: corev1.PullPolicy(viper.GetString(constant.KBImagePullPolicy)),
+ Command: []string{"kubectl"},
+ Args: append([]string{
+ "-n",
+ e.Namespace,
+ "exec",
+ e.PodName,
+ "-c",
+ e.Container,
+ "--",
+ }, e.Command...),
+ },
+ },
+ Volumes: []corev1.Volume{},
+ // tolerate all taints
+ Tolerations: []corev1.Toleration{
+ {
+ Operator: corev1.TolerationOpExists,
+ },
+ },
+ Affinity: &corev1.Affinity{},
+ NodeSelector: map[string]string{},
+ }
+}
+
+var _ Action = &ExecAction{}
diff --git a/internal/dataprotection/action/action_exec_test.go b/internal/dataprotection/action/action_exec_test.go
new file mode 100644
index 00000000000..dc35eea160d
--- /dev/null
+++ b/internal/dataprotection/action/action_exec_test.go
@@ -0,0 +1,127 @@
+/*
+Copyright (C) 2022-2023 ApeCloud Co., Ltd
+
+This file is part of KubeBlocks project
+
+This program is free software: you can redistribute it and/or modify
+it under the terms of the GNU Affero General Public License as published by
+the Free Software Foundation, either version 3 of the License, or
+(at your option) any later version.
+
+This program is distributed in the hope that it will be useful
+but WITHOUT ANY WARRANTY; without even the implied warranty of
+MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+GNU Affero General Public License for more details.
+
+You should have received a copy of the GNU Affero General Public License
+along with this program. If not, see .
+*/
+
+package action_test
+
+import (
+ . "github.com/onsi/ginkgo/v2"
+ . "github.com/onsi/gomega"
+
+ batchv1 "k8s.io/api/batch/v1"
+ metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
+ "sigs.k8s.io/controller-runtime/pkg/client"
+
+ dpv1alpha1 "github.com/apecloud/kubeblocks/apis/dataprotection/v1alpha1"
+ "github.com/apecloud/kubeblocks/internal/constant"
+ "github.com/apecloud/kubeblocks/internal/dataprotection/action"
+ "github.com/apecloud/kubeblocks/internal/generics"
+ testapps "github.com/apecloud/kubeblocks/internal/testutil/apps"
+ testdp "github.com/apecloud/kubeblocks/internal/testutil/dataprotection"
+ viper "github.com/apecloud/kubeblocks/internal/viperx"
+)
+
+var _ = Describe("ExecAction Test", func() {
+ const (
+ actionName = "test-exec-action"
+ podName = "pod"
+ container = "container"
+ serviceAccountName = "service-account"
+ )
+
+ var (
+ command = []string{"ls"}
+ )
+
+ cleanEnv := func() {
+ By("clean resources")
+ inNS := client.InNamespace(testCtx.DefaultNamespace)
+ testapps.ClearResourcesWithRemoveFinalizerOption(&testCtx, generics.BackupSignature, true, inNS)
+ testapps.ClearResourcesWithRemoveFinalizerOption(&testCtx, generics.JobSignature, true, inNS)
+ testapps.ClearResources(&testCtx, generics.PodSignature, inNS)
+ }
+
+ BeforeEach(func() {
+ cleanEnv()
+ viper.Set(constant.KBToolsImage, testdp.KBToolImage)
+ })
+
+ AfterEach(func() {
+ cleanEnv()
+ viper.Set(constant.KBToolsImage, "")
+ })
+
+ Context("create exec action", func() {
+ It("should return error when pod name is empty", func() {
+ act := &action.ExecAction{}
+ status, err := act.Execute(buildActionCtx())
+ Expect(err).To(HaveOccurred())
+ Expect(status).Should(BeNil())
+ })
+
+ It("should build pod spec but job action validate failed", func() {
+ act := &action.ExecAction{
+ JobAction: action.JobAction{
+ Name: actionName,
+ },
+ PodName: podName,
+ Namespace: testCtx.DefaultNamespace,
+ Command: command,
+ }
+ status, err := act.Execute(buildActionCtx())
+ Expect(err).To(HaveOccurred())
+ Expect(status).ShouldNot(BeNil())
+ Expect(status.Phase).Should(Equal(dpv1alpha1.ActionPhaseFailed))
+ Expect(act.JobAction.PodSpec).ShouldNot(BeNil())
+ })
+
+ It("should success to build exec action", func() {
+ labels := map[string]string{
+ "dp-test-action": actionName,
+ }
+
+ act := &action.ExecAction{
+ JobAction: action.JobAction{
+ Name: actionName,
+ ObjectMeta: metav1.ObjectMeta{
+ Name: actionName,
+ Namespace: testCtx.DefaultNamespace,
+ Labels: labels,
+ },
+ Owner: testdp.NewFakeBackup(&testCtx, nil),
+ },
+ PodName: podName,
+ Namespace: testCtx.DefaultNamespace,
+ Command: command,
+ Container: container,
+ ServiceAccountName: serviceAccountName,
+ }
+
+ By("should success to execute")
+ status, err := act.Execute(buildActionCtx())
+ Expect(err).Should(Succeed())
+ Expect(status).ShouldNot(BeNil())
+ Expect(status.Phase).Should(Equal(dpv1alpha1.ActionPhaseRunning))
+
+ By("check the job was created")
+ job := &batchv1.Job{}
+ key := client.ObjectKey{Name: actionName, Namespace: testCtx.DefaultNamespace}
+ Eventually(testapps.CheckObjExists(&testCtx, key, job, true)).Should(Succeed())
+ })
+ })
+})
diff --git a/internal/dataprotection/action/action_job.go b/internal/dataprotection/action/action_job.go
new file mode 100644
index 00000000000..931ccfb18d7
--- /dev/null
+++ b/internal/dataprotection/action/action_job.go
@@ -0,0 +1,137 @@
+/*
+Copyright (C) 2022-2023 ApeCloud Co., Ltd
+
+This file is part of KubeBlocks project
+
+This program is free software: you can redistribute it and/or modify
+it under the terms of the GNU Affero General Public License as published by
+the Free Software Foundation, either version 3 of the License, or
+(at your option) any later version.
+
+This program is distributed in the hope that it will be useful
+but WITHOUT ANY WARRANTY; without even the implied warranty of
+MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+GNU Affero General Public License for more details.
+
+You should have received a copy of the GNU Affero General Public License
+along with this program. If not, see .
+*/
+
+package action
+
+import (
+ "fmt"
+
+ batchv1 "k8s.io/api/batch/v1"
+ corev1 "k8s.io/api/core/v1"
+ metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
+ ref "k8s.io/client-go/tools/reference"
+ "sigs.k8s.io/controller-runtime/pkg/client"
+ "sigs.k8s.io/controller-runtime/pkg/controller/controllerutil"
+
+ dpv1alpha1 "github.com/apecloud/kubeblocks/apis/dataprotection/v1alpha1"
+ ctrlutil "github.com/apecloud/kubeblocks/internal/controllerutil"
+ "github.com/apecloud/kubeblocks/internal/dataprotection/types"
+ "github.com/apecloud/kubeblocks/internal/dataprotection/utils"
+)
+
+// JobAction is an action that creates a batch job.
+type JobAction struct {
+ // Name is the Name of the action.
+ Name string
+
+ // Owner is the owner of the job.
+ Owner client.Object
+
+ // ObjectMeta is the metadata of the job.
+ ObjectMeta metav1.ObjectMeta
+
+ // PodSpec is the
+ PodSpec *corev1.PodSpec
+
+ // BackOffLimit is the number of retries before considering a JobAction as failed.
+ BackOffLimit *int32
+}
+
+func (j *JobAction) GetName() string {
+ return j.Name
+}
+
+func (j *JobAction) Type() dpv1alpha1.ActionType {
+ return dpv1alpha1.ActionTypeJob
+}
+
+func (j *JobAction) Execute(ctx Context) (*dpv1alpha1.ActionStatus, error) {
+ sb := newStatusBuilder(j)
+ handleErr := func(err error) (*dpv1alpha1.ActionStatus, error) {
+ return sb.withErr(err).build(), err
+ }
+
+ if err := j.validate(); err != nil {
+ return handleErr(err)
+ }
+
+ key := client.ObjectKey{
+ Namespace: j.ObjectMeta.Namespace,
+ Name: j.ObjectMeta.Name,
+ }
+ original := batchv1.Job{}
+ exists, err := ctrlutil.CheckResourceExists(ctx.Ctx, ctx.Client, key, &original)
+ if err != nil {
+ return handleErr(err)
+ } else if exists {
+ // job exists, check job status and set action status accordingly
+ objRef, _ := ref.GetReference(ctx.Scheme, &original)
+ sb = sb.startTimestamp(&original.CreationTimestamp).objectRef(objRef)
+ _, finishedType, msg := utils.IsJobFinished(&original)
+ switch finishedType {
+ case batchv1.JobComplete:
+ return sb.phase(dpv1alpha1.ActionPhaseCompleted).
+ completionTimestamp(nil).
+ reason("").
+ build(), nil
+ case batchv1.JobFailed:
+ return sb.phase(dpv1alpha1.ActionPhaseFailed).
+ completionTimestamp(nil).
+ reason(msg).
+ build(), nil
+ }
+ // job is running
+ return sb.build(), nil
+ }
+
+ // job doesn't exist, create it
+ job := &batchv1.Job{
+ ObjectMeta: j.ObjectMeta,
+ Spec: batchv1.JobSpec{
+ Template: corev1.PodTemplateSpec{
+ ObjectMeta: j.ObjectMeta,
+ Spec: *j.PodSpec,
+ },
+ BackoffLimit: j.BackOffLimit,
+ },
+ }
+
+ controllerutil.AddFinalizer(job, types.DataProtectionFinalizerName)
+ if err = setControllerReference(j.Owner, job, ctx.Scheme); err != nil {
+ return handleErr(err)
+ }
+ msg := fmt.Sprintf("creating job %s/%s", job.Namespace, job.Name)
+ ctx.Recorder.Event(j.Owner, corev1.EventTypeNormal, "CreatingJob", msg)
+ return handleErr(client.IgnoreAlreadyExists(ctx.Client.Create(ctx.Ctx, job)))
+}
+
+func (j *JobAction) validate() error {
+ if j.ObjectMeta.Name == "" {
+ return fmt.Errorf("name is required")
+ }
+ if j.PodSpec == nil {
+ return fmt.Errorf("PodSpec is required")
+ }
+ if j.BackOffLimit == nil {
+ j.BackOffLimit = &types.DefaultBackOffLimit
+ }
+ return nil
+}
+
+var _ Action = &JobAction{}
diff --git a/internal/dataprotection/action/action_job_test.go b/internal/dataprotection/action/action_job_test.go
new file mode 100644
index 00000000000..f2604c096b1
--- /dev/null
+++ b/internal/dataprotection/action/action_job_test.go
@@ -0,0 +1,120 @@
+/*
+Copyright (C) 2022-2023 ApeCloud Co., Ltd
+
+This file is part of KubeBlocks project
+
+This program is free software: you can redistribute it and/or modify
+it under the terms of the GNU Affero General Public License as published by
+the Free Software Foundation, either version 3 of the License, or
+(at your option) any later version.
+
+This program is distributed in the hope that it will be useful
+but WITHOUT ANY WARRANTY; without even the implied warranty of
+MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+GNU Affero General Public License for more details.
+
+You should have received a copy of the GNU Affero General Public License
+along with this program. If not, see .
+*/
+
+package action_test
+
+import (
+ . "github.com/onsi/ginkgo/v2"
+ . "github.com/onsi/gomega"
+
+ batchv1 "k8s.io/api/batch/v1"
+ corev1 "k8s.io/api/core/v1"
+ metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
+ "sigs.k8s.io/controller-runtime/pkg/client"
+
+ dpv1alpha1 "github.com/apecloud/kubeblocks/apis/dataprotection/v1alpha1"
+ "github.com/apecloud/kubeblocks/internal/constant"
+ "github.com/apecloud/kubeblocks/internal/dataprotection/action"
+ "github.com/apecloud/kubeblocks/internal/generics"
+ testapps "github.com/apecloud/kubeblocks/internal/testutil/apps"
+ testdp "github.com/apecloud/kubeblocks/internal/testutil/dataprotection"
+ viper "github.com/apecloud/kubeblocks/internal/viperx"
+)
+
+var _ = Describe("JobAction Test", func() {
+ const (
+ actionName = "test-job-action"
+ container = "container"
+ )
+
+ var (
+ command = []string{"ls"}
+ )
+
+ cleanEnv := func() {
+ By("clean resources")
+ inNS := client.InNamespace(testCtx.DefaultNamespace)
+ testapps.ClearResourcesWithRemoveFinalizerOption(&testCtx, generics.JobSignature, true, inNS)
+ testapps.ClearResourcesWithRemoveFinalizerOption(&testCtx, generics.BackupSignature, true, inNS)
+ }
+
+ BeforeEach(func() {
+ cleanEnv()
+ viper.Set(constant.KBToolsImage, testdp.KBToolImage)
+ })
+
+ AfterEach(func() {
+ cleanEnv()
+ viper.Set(constant.KBToolsImage, "")
+ })
+
+ Context("create job action", func() {
+ It("should return error when pod spec is empty", func() {
+ act := &action.JobAction{}
+ status, err := act.Execute(buildActionCtx())
+ Expect(err).To(HaveOccurred())
+ Expect(status.Phase).Should(Equal(dpv1alpha1.ActionPhaseFailed))
+ })
+
+ It("should success to execute job action", func() {
+ labels := map[string]string{
+ "dp-test-action": actionName,
+ }
+
+ act := &action.JobAction{
+ Name: actionName,
+ ObjectMeta: metav1.ObjectMeta{
+ Name: actionName,
+ Namespace: testCtx.DefaultNamespace,
+ Labels: labels,
+ },
+ PodSpec: &corev1.PodSpec{
+ Containers: []corev1.Container{
+ {
+ Name: container,
+ Image: testdp.KBToolImage,
+ Command: command,
+ },
+ },
+ RestartPolicy: corev1.RestartPolicyNever,
+ },
+ Owner: testdp.NewFakeBackup(&testCtx, nil),
+ }
+
+ By("should success to execute")
+ status, err := act.Execute(buildActionCtx())
+ Expect(err).Should(Succeed())
+ Expect(status).ShouldNot(BeNil())
+ Expect(status.Phase).Should(Equal(dpv1alpha1.ActionPhaseRunning))
+
+ By("check the job was created")
+ job := &batchv1.Job{}
+ key := client.ObjectKey{Name: actionName, Namespace: testCtx.DefaultNamespace}
+ Eventually(testapps.CheckObjExists(&testCtx, key, job, true)).Should(Succeed())
+
+ By("set job status to complete")
+ testdp.PatchK8sJobStatus(&testCtx, client.ObjectKeyFromObject(job), batchv1.JobComplete)
+
+ By("action status should be completed")
+ status, err = act.Execute(buildActionCtx())
+ Expect(err).ShouldNot(HaveOccurred())
+ Expect(status.Phase).Should(Equal(dpv1alpha1.ActionPhaseCompleted))
+ })
+ })
+})
diff --git a/internal/dataprotection/action/builder_status.go b/internal/dataprotection/action/builder_status.go
new file mode 100644
index 00000000000..a249f04ec92
--- /dev/null
+++ b/internal/dataprotection/action/builder_status.go
@@ -0,0 +1,105 @@
+/*
+Copyright (C) 2022-2023 ApeCloud Co., Ltd
+
+This file is part of KubeBlocks project
+
+This program is free software: you can redistribute it and/or modify
+it under the terms of the GNU Affero General Public License as published by
+the Free Software Foundation, either version 3 of the License, or
+(at your option) any later version.
+
+This program is distributed in the hope that it will be useful
+but WITHOUT ANY WARRANTY; without even the implied warranty of
+MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+GNU Affero General Public License for more details.
+
+You should have received a copy of the GNU Affero General Public License
+along with this program. If not, see .
+*/
+
+package action
+
+import (
+ corev1 "k8s.io/api/core/v1"
+ metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
+
+ dpv1alpha1 "github.com/apecloud/kubeblocks/apis/dataprotection/v1alpha1"
+)
+
+type statusBuilder struct {
+ status *dpv1alpha1.ActionStatus
+}
+
+func newStatusBuilder(a Action) *statusBuilder {
+ sb := &statusBuilder{
+ status: &dpv1alpha1.ActionStatus{
+ Name: a.GetName(),
+ ActionType: a.Type(),
+ Phase: dpv1alpha1.ActionPhaseRunning,
+ },
+ }
+ return sb.startTimestamp(nil)
+}
+
+func (b *statusBuilder) phase(phase dpv1alpha1.ActionPhase) *statusBuilder {
+ b.status.Phase = phase
+ return b
+}
+
+func (b *statusBuilder) reason(reason string) *statusBuilder {
+ b.status.FailureReason = reason
+ return b
+}
+
+func (b *statusBuilder) startTimestamp(timestamp *metav1.Time) *statusBuilder {
+ t := timestamp
+ if t == nil {
+ t = &metav1.Time{
+ Time: metav1.Now().UTC(),
+ }
+ }
+ b.status.StartTimestamp = t
+ return b
+}
+
+func (b *statusBuilder) completionTimestamp(timestamp *metav1.Time) *statusBuilder {
+ t := timestamp
+ if t == nil {
+ t = &metav1.Time{
+ Time: metav1.Now().UTC(),
+ }
+ }
+ b.status.CompletionTimestamp = t
+ return b
+}
+
+func (b *statusBuilder) objectRef(objectRef *corev1.ObjectReference) *statusBuilder {
+ b.status.ObjectRef = objectRef
+ return b
+}
+
+func (b *statusBuilder) withErr(err error) *statusBuilder {
+ if err == nil {
+ return b
+ }
+ b.status.FailureReason = err.Error()
+ b.status.Phase = dpv1alpha1.ActionPhaseFailed
+ return b
+}
+
+func (b *statusBuilder) totalSize(size string) *statusBuilder {
+ b.status.TotalSize = size
+ return b
+}
+
+func (b *statusBuilder) timeRange(start, end *metav1.Time) *statusBuilder {
+ b.status.TimeRange = &dpv1alpha1.BackupTimeRange{
+ Start: start,
+ End: end,
+ }
+ return b
+}
+
+func (b *statusBuilder) build() *dpv1alpha1.ActionStatus {
+ return b.status
+}
diff --git a/internal/dataprotection/action/suite_test.go b/internal/dataprotection/action/suite_test.go
new file mode 100644
index 00000000000..2784861662c
--- /dev/null
+++ b/internal/dataprotection/action/suite_test.go
@@ -0,0 +1,152 @@
+/*
+Copyright (C) 2022-2023 ApeCloud Co., Ltd
+
+This file is part of KubeBlocks project
+
+This program is free software: you can redistribute it and/or modify
+it under the terms of the GNU Affero General Public License as published by
+the Free Software Foundation, either version 3 of the License, or
+(at your option) any later version.
+
+This program is distributed in the hope that it will be useful
+but WITHOUT ANY WARRANTY; without even the implied warranty of
+MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+GNU Affero General Public License for more details.
+
+You should have received a copy of the GNU Affero General Public License
+along with this program. If not, see .
+*/
+
+package action_test
+
+import (
+ "context"
+ "go/build"
+ "path/filepath"
+ "testing"
+
+ . "github.com/onsi/ginkgo/v2"
+ . "github.com/onsi/gomega"
+
+ "github.com/go-logr/logr"
+ vsv1 "github.com/kubernetes-csi/external-snapshotter/client/v6/apis/volumesnapshot/v1"
+ "go.uber.org/zap/zapcore"
+ "k8s.io/client-go/kubernetes/scheme"
+ "k8s.io/client-go/rest"
+ "k8s.io/client-go/tools/record"
+ ctrl "sigs.k8s.io/controller-runtime"
+ "sigs.k8s.io/controller-runtime/pkg/client"
+ "sigs.k8s.io/controller-runtime/pkg/envtest"
+ logf "sigs.k8s.io/controller-runtime/pkg/log"
+ "sigs.k8s.io/controller-runtime/pkg/log/zap"
+
+ appsv1alpha1 "github.com/apecloud/kubeblocks/apis/apps/v1alpha1"
+ dpv1alpha1 "github.com/apecloud/kubeblocks/apis/dataprotection/v1alpha1"
+ ctrlutil "github.com/apecloud/kubeblocks/internal/controllerutil"
+ "github.com/apecloud/kubeblocks/internal/dataprotection/action"
+ "github.com/apecloud/kubeblocks/internal/testutil"
+ viper "github.com/apecloud/kubeblocks/internal/viperx"
+)
+
+// These tests use Ginkgo (BDD-style Go testing framework). Refer to
+// http://onsi.github.io/ginkgo/ to learn more about Ginkgo.
+
+var (
+ cfg *rest.Config
+ k8sClient client.Client
+ testEnv *envtest.Environment
+ ctx context.Context
+ cancel context.CancelFunc
+ testCtx testutil.TestContext
+ logger logr.Logger
+ recorder record.EventRecorder
+
+ buildActionCtx = func() action.Context {
+ return action.Context{
+ Ctx: testCtx.Ctx,
+ Client: testCtx.Cli,
+ Recorder: recorder,
+ Scheme: testEnv.Scheme,
+ RestClientConfig: cfg,
+ }
+ }
+)
+
+func init() {
+ viper.AutomaticEnv()
+}
+
+func TestAction(t *testing.T) {
+ RegisterFailHandler(Fail)
+
+ RunSpecs(t, "Data Protection Action Suite")
+}
+
+var _ = BeforeSuite(func() {
+ if viper.GetBool("ENABLE_DEBUG_LOG") {
+ logf.SetLogger(zap.New(zap.WriteTo(GinkgoWriter), zap.UseDevMode(true), func(o *zap.Options) {
+ o.TimeEncoder = zapcore.ISO8601TimeEncoder
+ }))
+ }
+
+ ctx, cancel = context.WithCancel(context.TODO())
+ logger = logf.FromContext(ctx).WithValues()
+ logger.Info("logger start")
+
+ By("bootstrapping test environment")
+ testEnv = &envtest.Environment{
+ CRDDirectoryPaths: []string{
+ filepath.Join("..", "..", "..", "config", "crd", "bases"),
+ // use dependent external crds.
+ // resolved by ref: https://github.com/operator-framework/operator-sdk/issues/4434#issuecomment-786794418
+ filepath.Join(build.Default.GOPATH, "pkg", "mod", "github.com", "kubernetes-csi/external-snapshotter/",
+ "client/v6@v6.2.0", "config", "crd"),
+ },
+ ErrorIfCRDPathMissing: true,
+ }
+
+ var err error
+ // cfg is defined in this file globally.
+ cfg, err = testEnv.Start()
+ Expect(err).NotTo(HaveOccurred())
+ Expect(cfg).NotTo(BeNil())
+
+ err = appsv1alpha1.AddToScheme(scheme.Scheme)
+ Expect(err).NotTo(HaveOccurred())
+
+ err = vsv1.AddToScheme(scheme.Scheme)
+ Expect(err).NotTo(HaveOccurred())
+
+ err = dpv1alpha1.AddToScheme(scheme.Scheme)
+ Expect(err).NotTo(HaveOccurred())
+
+ // +kubebuilder:scaffold:scheme
+
+ k8sClient, err = client.New(cfg, client.Options{Scheme: scheme.Scheme})
+ Expect(err).NotTo(HaveOccurred())
+ Expect(k8sClient).NotTo(BeNil())
+
+ // run reconcile
+ k8sManager, err := ctrl.NewManager(cfg, ctrl.Options{
+ Scheme: scheme.Scheme,
+ MetricsBindAddress: "0",
+ ClientDisableCacheFor: ctrlutil.GetUncachedObjects(),
+ })
+ Expect(err).ToNot(HaveOccurred())
+
+ testCtx = testutil.NewDefaultTestContext(ctx, k8sClient, testEnv)
+ recorder = k8sManager.GetEventRecorderFor("dataprotection-action-test")
+
+ go func() {
+ defer GinkgoRecover()
+ err = k8sManager.Start(ctx)
+ Expect(err).ToNot(HaveOccurred(), "failed to run manager")
+ }()
+})
+
+var _ = AfterSuite(func() {
+ cancel()
+ By("tearing down the test environment")
+ err := testEnv.Stop()
+ Expect(err).NotTo(HaveOccurred())
+})
diff --git a/internal/dataprotection/action/types.go b/internal/dataprotection/action/types.go
new file mode 100644
index 00000000000..ddc2d4db0d2
--- /dev/null
+++ b/internal/dataprotection/action/types.go
@@ -0,0 +1,27 @@
+/*
+Copyright (C) 2022-2023 ApeCloud Co., Ltd
+
+This file is part of KubeBlocks project
+
+This program is free software: you can redistribute it and/or modify
+it under the terms of the GNU Affero General Public License as published by
+the Free Software Foundation, either version 3 of the License, or
+(at your option) any later version.
+
+This program is distributed in the hope that it will be useful
+but WITHOUT ANY WARRANTY; without even the implied warranty of
+MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+GNU Affero General Public License for more details.
+
+You should have received a copy of the GNU Affero General Public License
+along with this program. If not, see .
+*/
+
+package action
+
+type ErrorMode string
+
+const (
+ ErrorModeContinue ErrorMode = "Continue"
+ ErrorModeFail ErrorMode = "Fail"
+)
diff --git a/internal/webhook/webhook.go b/internal/dataprotection/action/utils.go
similarity index 53%
rename from internal/webhook/webhook.go
rename to internal/dataprotection/action/utils.go
index 068d891be5d..c78fcd9966d 100644
--- a/internal/webhook/webhook.go
+++ b/internal/dataprotection/action/utils.go
@@ -17,27 +17,19 @@ You should have received a copy of the GNU Affero General Public License
along with this program. If not, see .
*/
-package webhook
+package action
import (
- ctrl "sigs.k8s.io/controller-runtime"
- "sigs.k8s.io/controller-runtime/pkg/manager"
- "sigs.k8s.io/controller-runtime/pkg/webhook"
- "sigs.k8s.io/controller-runtime/pkg/webhook/admission"
-)
+ "reflect"
-var (
- setupLog = ctrl.Log.WithName("webhook-setup")
- // HandlerMap contains all admission webhook handlers.
- HandlerMap = map[string]admission.Handler{}
+ metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
+ "k8s.io/apimachinery/pkg/runtime"
+ ctrlutil "sigs.k8s.io/controller-runtime/pkg/controller/controllerutil"
)
-func SetupWithManager(mgr manager.Manager) error {
- server := mgr.GetWebhookServer()
- // register admission handlers
- for path, handler := range HandlerMap {
- server.Register(path, &webhook.Admission{Handler: handler})
- setupLog.Info("Registered webhook handler", "path", path)
+func setControllerReference(owner, controlled metav1.Object, scheme *runtime.Scheme) error {
+ if owner == nil || reflect.ValueOf(owner).IsNil() {
+ return nil
}
- return nil
+ return ctrlutil.SetControllerReference(owner, controlled, scheme)
}
diff --git a/internal/dataprotection/backup/deleter.go b/internal/dataprotection/backup/deleter.go
new file mode 100644
index 00000000000..a0d50a522d9
--- /dev/null
+++ b/internal/dataprotection/backup/deleter.go
@@ -0,0 +1,252 @@
+/*
+Copyright (C) 2022-2023 ApeCloud Co., Ltd
+
+This file is part of KubeBlocks project
+
+This program is free software: you can redistribute it and/or modify
+it under the terms of the GNU Affero General Public License as published by
+the Free Software Foundation, either version 3 of the License, or
+(at your option) any later version.
+
+This program is distributed in the hope that it will be useful
+but WITHOUT ANY WARRANTY; without even the implied warranty of
+MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+GNU Affero General Public License for more details.
+
+You should have received a copy of the GNU Affero General Public License
+along with this program. If not, see .
+*/
+
+package backup
+
+import (
+ "fmt"
+ "strings"
+
+ vsv1 "github.com/kubernetes-csi/external-snapshotter/client/v6/apis/volumesnapshot/v1"
+ batchv1 "k8s.io/api/batch/v1"
+ corev1 "k8s.io/api/core/v1"
+ apierrors "k8s.io/apimachinery/pkg/api/errors"
+ metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
+ "k8s.io/apimachinery/pkg/runtime"
+ "k8s.io/apimachinery/pkg/types"
+ "sigs.k8s.io/controller-runtime/pkg/client"
+ "sigs.k8s.io/controller-runtime/pkg/controller/controllerutil"
+
+ dpv1alpha1 "github.com/apecloud/kubeblocks/apis/dataprotection/v1alpha1"
+ "github.com/apecloud/kubeblocks/internal/constant"
+ ctrlutil "github.com/apecloud/kubeblocks/internal/controllerutil"
+ dptypes "github.com/apecloud/kubeblocks/internal/dataprotection/types"
+ "github.com/apecloud/kubeblocks/internal/dataprotection/utils"
+ "github.com/apecloud/kubeblocks/internal/dataprotection/utils/boolptr"
+ viper "github.com/apecloud/kubeblocks/internal/viperx"
+)
+
+const (
+ deleteBackupFilesJobNamePrefix = "delete-"
+)
+
+type DeletionStatus string
+
+const (
+ DeletionStatusDeleting DeletionStatus = "Deleting"
+ DeletionStatusFailed DeletionStatus = "Failed"
+ DeletionStatusSucceeded DeletionStatus = "Succeeded"
+ DeletionStatusUnknown DeletionStatus = "Unknown"
+)
+
+type Deleter struct {
+ ctrlutil.RequestCtx
+ Client client.Client
+ Scheme *runtime.Scheme
+}
+
+func (d *Deleter) DeleteBackupFiles(backup *dpv1alpha1.Backup) (DeletionStatus, error) {
+ jobKey := BuildDeleteBackupFilesJobKey(backup)
+ job := &batchv1.Job{}
+ exists, err := ctrlutil.CheckResourceExists(d.Ctx, d.Client, jobKey, job)
+ if err != nil {
+ return DeletionStatusUnknown, err
+ }
+
+ // if deletion job exists, check its status
+ if exists {
+ _, finishedType, msg := utils.IsJobFinished(job)
+ switch finishedType {
+ case batchv1.JobComplete:
+ return DeletionStatusSucceeded, nil
+ case batchv1.JobFailed:
+ return DeletionStatusFailed,
+ fmt.Errorf("deletion backup files job \"%s\" failed, you can delete it to re-delete the backup files, %s", job.Name, msg)
+ }
+ return DeletionStatusDeleting, nil
+ }
+
+ // if deletion job not exists, create it
+ pvcName := backup.Status.PersistentVolumeClaimName
+ if pvcName == "" {
+ d.Log.Info("skip deleting backup files because PersistentVolumeClaimName is empty",
+ "backup", backup.Name)
+ return DeletionStatusSucceeded, nil
+ }
+
+ // check if backup repo PVC exists, if not, skip to delete backup files
+ pvcKey := client.ObjectKey{Namespace: backup.Namespace, Name: pvcName}
+ if err = d.Client.Get(d.Ctx, pvcKey, &corev1.PersistentVolumeClaim{}); err != nil {
+ if apierrors.IsNotFound(err) {
+ return DeletionStatusSucceeded, nil
+ }
+ return DeletionStatusUnknown, err
+ }
+
+ backupFilePath := backup.Status.Path
+ if backupFilePath == "" || !strings.Contains(backupFilePath, backup.Name) {
+ // For compatibility: the FilePath field is changing from time to time,
+ // and it may not contain the backup name as a path component if the Backup object
+ // was created in a previous version. In this case, it's dangerous to execute
+ // the deletion command. For example, files belongs to other Backups can be deleted as well.
+ d.Log.Info("skip deleting backup files because backup file path is invalid",
+ "backupFilePath", backupFilePath, "backup", backup.Name)
+ return DeletionStatusSucceeded, nil
+ }
+ return DeletionStatusDeleting, d.createDeleteBackupFileJob(jobKey, backup, pvcName, backup.Status.Path)
+}
+
+func (d *Deleter) createDeleteBackupFileJob(
+ jobKey types.NamespacedName,
+ backup *dpv1alpha1.Backup,
+ backupPVCName string,
+ backupFilePath string) error {
+ // make sure the path has a leading slash
+ if !strings.HasPrefix(backupFilePath, "/") {
+ backupFilePath = "/" + backupFilePath
+ }
+
+ // this script first deletes the directory where the backup is located (including files
+ // in the directory), and then traverses up the path level by level to clean up empty directories.
+ deleteScript := fmt.Sprintf(`
+ backupPathBase=%s;
+ targetPath="${backupPathBase}%s";
+
+ echo "removing backup files in ${targetPath}";
+ rm -rf "${targetPath}";
+
+ absBackupPathBase=$(realpath "${backupPathBase}");
+ curr=$(realpath "${targetPath}");
+ while true; do
+ parent=$(dirname "${curr}");
+ if [ "${parent}" == "${absBackupPathBase}" ]; then
+ echo "reach backupPathBase ${backupPathBase}, done";
+ break;
+ fi;
+ if [ ! "$(ls -A "${parent}")" ]; then
+ echo "${parent} is empty, removing it...";
+ rmdir "${parent}";
+ else
+ echo "${parent} is not empty, done";
+ break;
+ fi;
+ curr="${parent}";
+ done
+ `, RepoVolumeMountPath, backupFilePath)
+
+ runAsUser := int64(0)
+ container := corev1.Container{
+ Name: backup.Name,
+ Command: []string{"sh", "-c"},
+ Args: []string{deleteScript},
+ Image: viper.GetString(constant.KBToolsImage),
+ ImagePullPolicy: corev1.PullPolicy(viper.GetString(constant.KBImagePullPolicy)),
+ SecurityContext: &corev1.SecurityContext{
+ AllowPrivilegeEscalation: boolptr.False(),
+ RunAsUser: &runAsUser,
+ },
+ VolumeMounts: []corev1.VolumeMount{
+ buildBackupRepoVolumeMount(backupPVCName),
+ },
+ }
+ ctrlutil.InjectZeroResourcesLimitsIfEmpty(&container)
+
+ // build pod
+ podSpec := corev1.PodSpec{
+ Containers: []corev1.Container{container},
+ RestartPolicy: corev1.RestartPolicyNever,
+ Volumes: []corev1.Volume{
+ buildBackupRepoVolume(backupPVCName),
+ },
+ }
+
+ if err := utils.AddTolerations(&podSpec); err != nil {
+ return err
+ }
+
+ // build job
+ job := &batchv1.Job{
+ ObjectMeta: metav1.ObjectMeta{
+ Namespace: jobKey.Namespace,
+ Name: jobKey.Name,
+ },
+ Spec: batchv1.JobSpec{
+ Template: corev1.PodTemplateSpec{
+ ObjectMeta: metav1.ObjectMeta{
+ Namespace: jobKey.Namespace,
+ Name: jobKey.Name,
+ },
+ Spec: podSpec,
+ },
+ BackoffLimit: &dptypes.DefaultBackOffLimit,
+ },
+ }
+ if err := controllerutil.SetControllerReference(backup, job, d.Scheme); err != nil {
+ return err
+ }
+ d.Log.V(1).Info("create a job to delete backup files", "job", job)
+ return client.IgnoreAlreadyExists(d.Client.Create(d.Ctx, job))
+}
+
+func (d *Deleter) DeleteVolumeSnapshots(backup *dpv1alpha1.Backup) error {
+ // initialize volume snapshot client that is compatible with both v1beta1 and v1
+ vsCli := &ctrlutil.VolumeSnapshotCompatClient{
+ Client: d.Client,
+ Ctx: d.Ctx,
+ }
+
+ snaps := &vsv1.VolumeSnapshotList{}
+ if err := vsCli.List(snaps, client.InNamespace(backup.Namespace),
+ client.MatchingLabels(BuildBackupWorkloadLabels(backup))); err != nil {
+ return client.IgnoreNotFound(err)
+ }
+
+ deleteVolumeSnapshot := func(vs *vsv1.VolumeSnapshot) error {
+ if controllerutil.ContainsFinalizer(vs, dptypes.DataProtectionFinalizerName) {
+ patch := vs.DeepCopy()
+ controllerutil.RemoveFinalizer(vs, dptypes.DataProtectionFinalizerName)
+ if err := vsCli.Patch(vs, patch); err != nil {
+ return err
+ }
+ }
+ if !vs.DeletionTimestamp.IsZero() {
+ return nil
+ }
+ d.Log.V(1).Info("delete volume snapshot", "volume snapshot", vs)
+ if err := vsCli.Delete(vs); err != nil {
+ return err
+ }
+ return nil
+ }
+
+ for i := range snaps.Items {
+ if err := deleteVolumeSnapshot(&snaps.Items[i]); err != nil {
+ return err
+ }
+ }
+ return nil
+}
+
+func BuildDeleteBackupFilesJobKey(backup *dpv1alpha1.Backup) client.ObjectKey {
+ jobName := fmt.Sprintf("%s-%s%s", backup.UID[:8], deleteBackupFilesJobNamePrefix, backup.Name)
+ if len(jobName) > 63 {
+ jobName = jobName[:63]
+ }
+ return client.ObjectKey{Namespace: backup.Namespace, Name: jobName}
+}
diff --git a/internal/dataprotection/backup/deleter_test.go b/internal/dataprotection/backup/deleter_test.go
new file mode 100644
index 00000000000..b4988c3ece3
--- /dev/null
+++ b/internal/dataprotection/backup/deleter_test.go
@@ -0,0 +1,160 @@
+/*
+Copyright (C) 2022-2023 ApeCloud Co., Ltd
+
+This file is part of KubeBlocks project
+
+This program is free software: you can redistribute it and/or modify
+it under the terms of the GNU Affero General Public License as published by
+the Free Software Foundation, either version 3 of the License, or
+(at your option) any later version.
+
+This program is distributed in the hope that it will be useful
+but WITHOUT ANY WARRANTY; without even the implied warranty of
+MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+GNU Affero General Public License for more details.
+
+You should have received a copy of the GNU Affero General Public License
+along with this program. If not, see .
+*/
+
+package backup
+
+import (
+ . "github.com/onsi/ginkgo/v2"
+ . "github.com/onsi/gomega"
+
+ batchv1 "k8s.io/api/batch/v1"
+ "sigs.k8s.io/controller-runtime/pkg/client"
+
+ dpv1alpha1 "github.com/apecloud/kubeblocks/apis/dataprotection/v1alpha1"
+ "github.com/apecloud/kubeblocks/internal/constant"
+ ctrlutil "github.com/apecloud/kubeblocks/internal/controllerutil"
+ "github.com/apecloud/kubeblocks/internal/generics"
+ testapps "github.com/apecloud/kubeblocks/internal/testutil/apps"
+ testdp "github.com/apecloud/kubeblocks/internal/testutil/dataprotection"
+ viper "github.com/apecloud/kubeblocks/internal/viperx"
+)
+
+var _ = Describe("Backup Deleter Test", func() {
+ const (
+ backupRepoPVCName = "backup-repo-pvc"
+ backupPath = "/backup/test-backup"
+ backupVSName = "backup-vs"
+ backupPVCName = "backup-pvc"
+ )
+
+ buildDeleter := func() *Deleter {
+ return &Deleter{
+ RequestCtx: ctrlutil.RequestCtx{
+ Log: logger,
+ Ctx: testCtx.Ctx,
+ Recorder: recorder,
+ },
+ Scheme: testEnv.Scheme,
+ Client: testCtx.Cli,
+ }
+ }
+
+ cleanEnv := func() {
+ By("clean resources")
+ inNS := client.InNamespace(testCtx.DefaultNamespace)
+ testapps.ClearResourcesWithRemoveFinalizerOption(&testCtx, generics.BackupSignature, true, inNS)
+ testapps.ClearResourcesWithRemoveFinalizerOption(&testCtx, generics.JobSignature, true, inNS)
+ testapps.ClearResources(&testCtx, generics.VolumeSnapshotSignature, inNS)
+ }
+
+ BeforeEach(func() {
+ cleanEnv()
+ viper.Set(constant.KBToolsImage, testdp.KBToolImage)
+ })
+
+ AfterEach(func() {
+ cleanEnv()
+ viper.Set(constant.KBToolsImage, "")
+ })
+
+ Context("delete backup file", func() {
+ var (
+ backup *dpv1alpha1.Backup
+ deleter *Deleter
+ )
+
+ BeforeEach(func() {
+ backup = testdp.NewFakeBackup(&testCtx, nil)
+ deleter = buildDeleter()
+ })
+
+ It("should success when backup status PVC is empty", func() {
+ Expect(backup.Status.PersistentVolumeClaimName).Should(Equal(""))
+ status, err := deleter.DeleteBackupFiles(backup)
+ Expect(err).ShouldNot(HaveOccurred())
+ Expect(status).Should(Equal(DeletionStatusSucceeded))
+ })
+
+ It("should success when backup status path is empty", func() {
+ backup.Status.PersistentVolumeClaimName = backupRepoPVCName
+ Expect(backup.Status.Path).Should(Equal(""))
+ status, err := deleter.DeleteBackupFiles(backup)
+ Expect(err).ShouldNot(HaveOccurred())
+ Expect(status).Should(Equal(DeletionStatusSucceeded))
+ })
+
+ It("should success when PVC does not exist", func() {
+ backup.Status.PersistentVolumeClaimName = backupRepoPVCName
+ backup.Status.Path = backupPath
+ status, err := deleter.DeleteBackupFiles(backup)
+ Expect(err).ShouldNot(HaveOccurred())
+ Expect(status).Should(Equal(DeletionStatusSucceeded))
+ })
+
+ It("should create job to delete backup file", func() {
+ By("mock backup repo PVC")
+ backupRepoPVC := testdp.NewFakePVC(&testCtx, backupRepoPVCName)
+
+ By("delete backup file")
+ backup.Status.PersistentVolumeClaimName = backupRepoPVC.Name
+ backup.Status.Path = backupPath
+ status, err := deleter.DeleteBackupFiles(backup)
+ Expect(err).ShouldNot(HaveOccurred())
+ Expect(status).Should(Equal(DeletionStatusDeleting))
+
+ By("check job exist")
+ job := &batchv1.Job{}
+ key := BuildDeleteBackupFilesJobKey(backup)
+ Eventually(testapps.CheckObjExists(&testCtx, key, job, true)).Should(Succeed())
+ })
+ })
+
+ Context("delete volume snapshots", func() {
+ var (
+ backup *dpv1alpha1.Backup
+ deleter *Deleter
+ )
+
+ BeforeEach(func() {
+ backup = testdp.NewFakeBackup(&testCtx, nil)
+ deleter = buildDeleter()
+ })
+
+ It("should success when volume snapshot does not exist", func() {
+ Expect(deleter.DeleteVolumeSnapshots(backup)).Should(Succeed())
+ })
+
+ It("should success when volume snapshot exist", func() {
+ By("mock volume snapshot")
+ vs := testdp.NewVolumeSnapshotFactory(testCtx.DefaultNamespace, backupVSName).
+ SetSourcePVCName(backupPVCName).
+ AddLabelsInMap(BuildBackupWorkloadLabels(backup)).
+ Create(&testCtx).GetObject()
+ Eventually(testapps.CheckObjExists(&testCtx,
+ client.ObjectKeyFromObject(vs), vs, true)).Should(Succeed())
+
+ By("delete volume snapshot")
+ Expect(deleter.DeleteVolumeSnapshots(backup)).Should(Succeed())
+
+ By("check volume snapshot deleted")
+ Eventually(testapps.CheckObjExists(&testCtx,
+ client.ObjectKeyFromObject(vs), vs, false)).Should(Succeed())
+ })
+ })
+})
diff --git a/internal/dataprotection/backup/request.go b/internal/dataprotection/backup/request.go
new file mode 100644
index 00000000000..0a9cf970a6e
--- /dev/null
+++ b/internal/dataprotection/backup/request.go
@@ -0,0 +1,410 @@
+/*
+Copyright (C) 2022-2023 ApeCloud Co., Ltd
+
+This file is part of KubeBlocks project
+
+This program is free software: you can redistribute it and/or modify
+it under the terms of the GNU Affero General Public License as published by
+the Free Software Foundation, either version 3 of the License, or
+(at your option) any later version.
+
+This program is distributed in the hope that it will be useful
+but WITHOUT ANY WARRANTY; without even the implied warranty of
+MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+GNU Affero General Public License for more details.
+
+You should have received a copy of the GNU Affero General Public License
+along with this program. If not, see .
+*/
+
+package backup
+
+import (
+ "fmt"
+ "reflect"
+
+ corev1 "k8s.io/api/core/v1"
+ metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
+ "sigs.k8s.io/controller-runtime/pkg/client"
+
+ dpv1alpha1 "github.com/apecloud/kubeblocks/apis/dataprotection/v1alpha1"
+ "github.com/apecloud/kubeblocks/internal/constant"
+ intctrlutil "github.com/apecloud/kubeblocks/internal/controllerutil"
+ "github.com/apecloud/kubeblocks/internal/dataprotection/action"
+ dptypes "github.com/apecloud/kubeblocks/internal/dataprotection/types"
+ "github.com/apecloud/kubeblocks/internal/dataprotection/utils"
+ "github.com/apecloud/kubeblocks/internal/dataprotection/utils/boolptr"
+ viper "github.com/apecloud/kubeblocks/internal/viperx"
+)
+
+const (
+ BackupDataJobNamePrefix = "dp-backup"
+ prebackupJobNamePrefix = "dp-prebackup"
+ postbackupJobNamePrefix = "dp-postbackup"
+ backupDataContainerName = "backupdata"
+ syncProgressContainerName = "sync-progress"
+)
+
+// Request is a request for a backup, with all references to other objects.
+type Request struct {
+ *dpv1alpha1.Backup
+ intctrlutil.RequestCtx
+
+ Client client.Client
+ BackupPolicy *dpv1alpha1.BackupPolicy
+ BackupMethod *dpv1alpha1.BackupMethod
+ ActionSet *dpv1alpha1.ActionSet
+ TargetPods []*corev1.Pod
+ BackupRepoPVC *corev1.PersistentVolumeClaim
+ BackupRepo *dpv1alpha1.BackupRepo
+}
+
+func (r *Request) GetBackupType() string {
+ if r.ActionSet != nil {
+ return string(r.ActionSet.Spec.BackupType)
+ }
+ if r.BackupMethod != nil && boolptr.IsSetToTrue(r.BackupMethod.SnapshotVolumes) {
+ return string(dpv1alpha1.BackupTypeFull)
+ }
+ return ""
+}
+
+// BuildActions builds the actions for the backup.
+func (r *Request) BuildActions() ([]action.Action, error) {
+ var actions []action.Action
+
+ appendIgnoreNil := func(elems ...action.Action) {
+ for _, elem := range elems {
+ if elem == nil || reflect.ValueOf(elem).IsNil() {
+ continue
+ }
+ actions = append(actions, elem)
+ }
+ }
+
+ // build pre-backup actions
+ preBackupActions, err := r.buildPreBackupActions()
+ if err != nil {
+ return nil, err
+ }
+
+ // build backup data action
+ backupDataAction, err := r.buildBackupDataAction()
+ if err != nil {
+ return nil, err
+ }
+
+ // build create volume snapshot action
+ createVolumeSnapshotAction, err := r.buildCreateVolumeSnapshotAction()
+ if err != nil {
+ return nil, err
+ }
+
+ // build backup kubernetes resources action
+ backupKubeResourcesAction, err := r.buildBackupKubeResourcesAction()
+ if err != nil {
+ return nil, err
+ }
+
+ // build post-backup actions
+ postBackupActions, err := r.buildPostBackupActions()
+ if err != nil {
+ return nil, err
+ }
+
+ appendIgnoreNil(preBackupActions...)
+ appendIgnoreNil(backupDataAction, createVolumeSnapshotAction, backupKubeResourcesAction)
+ appendIgnoreNil(postBackupActions...)
+ return actions, nil
+}
+
+func (r *Request) buildPreBackupActions() ([]action.Action, error) {
+ if !r.backupActionSetExists() ||
+ len(r.ActionSet.Spec.Backup.PreBackup) == 0 {
+ return nil, nil
+ }
+
+ var actions []action.Action
+ for i, preBackup := range r.ActionSet.Spec.Backup.PreBackup {
+ a, err := r.buildAction(fmt.Sprintf("%s-%d", prebackupJobNamePrefix, i), &preBackup)
+ if err != nil {
+ return nil, err
+ }
+ actions = append(actions, a)
+ }
+ return actions, nil
+}
+
+func (r *Request) buildPostBackupActions() ([]action.Action, error) {
+ if !r.backupActionSetExists() ||
+ len(r.ActionSet.Spec.Backup.PostBackup) == 0 {
+ return nil, nil
+ }
+
+ var actions []action.Action
+ for i, postBackup := range r.ActionSet.Spec.Backup.PostBackup {
+ a, err := r.buildAction(fmt.Sprintf("%s-%d", postbackupJobNamePrefix, i), &postBackup)
+ if err != nil {
+ return nil, err
+ }
+ actions = append(actions, a)
+ }
+ return actions, nil
+}
+
+func (r *Request) buildBackupDataAction() (action.Action, error) {
+ if !r.backupActionSetExists() ||
+ r.ActionSet.Spec.Backup.BackupData == nil {
+ return nil, nil
+ }
+
+ backupDataAct := r.ActionSet.Spec.Backup.BackupData
+ podSpec := r.buildJobActionPodSpec(backupDataContainerName, &backupDataAct.JobActionSpec)
+ if backupDataAct.SyncProgress != nil {
+ r.injectSyncProgressContainer(podSpec, backupDataAct.SyncProgress)
+ }
+
+ if r.ActionSet.Spec.BackupType == dpv1alpha1.BackupTypeFull {
+ return &action.JobAction{
+ Name: BackupDataJobNamePrefix,
+ ObjectMeta: *buildBackupJobObjMeta(r.Backup, BackupDataJobNamePrefix),
+ Owner: r.Backup,
+ PodSpec: podSpec,
+ BackOffLimit: r.BackupPolicy.Spec.BackoffLimit,
+ }, nil
+ }
+ return nil, fmt.Errorf("unsupported backup type %s", r.ActionSet.Spec.BackupType)
+}
+
+func (r *Request) buildCreateVolumeSnapshotAction() (action.Action, error) {
+ targetPod := r.TargetPods[0]
+ if r.BackupMethod == nil ||
+ !boolptr.IsSetToTrue(r.BackupMethod.SnapshotVolumes) {
+ return nil, nil
+ }
+
+ if r.BackupMethod.TargetVolumes == nil {
+ return nil, fmt.Errorf("targetVolumes is required for snapshotVolumes")
+ }
+
+ if !utils.VolumeSnapshotEnabled() {
+ return nil, fmt.Errorf("volume snapshot is not enabled")
+ }
+
+ pvcs, err := getPVCsByVolumeNames(r.Client, targetPod, r.BackupMethod.TargetVolumes.Volumes)
+ if err != nil {
+ return nil, err
+ }
+
+ if len(pvcs) == 0 {
+ return nil, fmt.Errorf("no PVCs found for pod %s to back up", targetPod.Name)
+ }
+
+ return &action.CreateVolumeSnapshotAction{
+ Name: "createVolumeSnapshot",
+ ObjectMeta: metav1.ObjectMeta{
+ Namespace: r.Backup.Namespace,
+ Name: r.Backup.Name,
+ Labels: BuildBackupWorkloadLabels(r.Backup),
+ },
+ Owner: r.Backup,
+ PersistentVolumeClaimWrappers: pvcs,
+ }, nil
+}
+
+// TODO(ldm): implement this
+func (r *Request) buildBackupKubeResourcesAction() (action.Action, error) {
+ return nil, nil
+}
+
+func (r *Request) buildAction(name string, act *dpv1alpha1.ActionSpec) (action.Action, error) {
+ if act.Exec == nil && act.Job == nil {
+ return nil, fmt.Errorf("action %s has no exec or job", name)
+ }
+ if act.Exec != nil && act.Job != nil {
+ return nil, fmt.Errorf("action %s should have only one of exec or job", name)
+ }
+ switch {
+ case act.Exec != nil:
+ return r.buildExecAction(name, act.Exec), nil
+ case act.Job != nil:
+ return r.buildJobAction(name, act.Job)
+ }
+ return nil, nil
+}
+
+func (r *Request) buildExecAction(name string, exec *dpv1alpha1.ExecActionSpec) action.Action {
+ targetPod := r.TargetPods[0]
+ return &action.ExecAction{
+ JobAction: action.JobAction{
+ Name: name,
+ ObjectMeta: *buildBackupJobObjMeta(r.Backup, name),
+ Owner: r.Backup,
+ },
+ Command: exec.Command,
+ Container: exec.Container,
+ Namespace: targetPod.Namespace,
+ PodName: targetPod.Name,
+ Timeout: exec.Timeout,
+ ServiceAccountName: r.targetServiceAccountName(),
+ }
+}
+
+func (r *Request) buildJobAction(name string, job *dpv1alpha1.JobActionSpec) (action.Action, error) {
+ return &action.JobAction{
+ Name: name,
+ ObjectMeta: *buildBackupJobObjMeta(r.Backup, name),
+ Owner: r.Backup,
+ PodSpec: r.buildJobActionPodSpec(name, job),
+ BackOffLimit: r.BackupPolicy.Spec.BackoffLimit,
+ }, nil
+}
+
+func (r *Request) buildJobActionPodSpec(name string, job *dpv1alpha1.JobActionSpec) *corev1.PodSpec {
+ targetPod := r.TargetPods[0]
+ // build environment variables, include built-in envs, envs from backupMethod
+ // and envs from actionSet. Latter will override former for the same name.
+ // env from backupMethod has the highest priority.
+ buildEnv := func() []corev1.EnvVar {
+ envVars := []corev1.EnvVar{
+ {
+ Name: dptypes.DPBackupName,
+ Value: r.Backup.Name,
+ },
+ {
+ Name: dptypes.DPBackupDIR,
+ Value: buildBackupPathInContainer(r.Backup, r.BackupPolicy.Spec.PathPrefix),
+ },
+ {
+ Name: dptypes.DPTargetPodName,
+ Value: targetPod.Name,
+ },
+ {
+ Name: dptypes.DPTTL,
+ Value: r.Spec.RetentionPeriod.String(),
+ },
+ }
+ envVars = append(envVars, utils.BuildEnvByCredential(targetPod, r.BackupPolicy.Spec.Target.ConnectionCredential)...)
+ if r.ActionSet != nil {
+ envVars = append(envVars, r.ActionSet.Spec.Env...)
+ }
+ return utils.MergeEnv(envVars, r.BackupMethod.Env)
+ }
+
+ buildVolumes := func() []corev1.Volume {
+ return append([]corev1.Volume{
+ buildBackupRepoVolume(r.BackupRepoPVC.Name),
+ }, getVolumesByVolumeInfo(targetPod, r.BackupMethod.TargetVolumes)...)
+ }
+
+ buildVolumeMounts := func() []corev1.VolumeMount {
+ return append([]corev1.VolumeMount{
+ buildBackupRepoVolumeMount(r.BackupRepoPVC.Name),
+ }, getVolumeMountsByVolumeInfo(targetPod, r.BackupMethod.TargetVolumes)...)
+ }
+
+ runAsUser := int64(0)
+ container := corev1.Container{
+ Name: name,
+ Image: job.Image,
+ Command: job.Command,
+ Env: buildEnv(),
+ VolumeMounts: buildVolumeMounts(),
+ ImagePullPolicy: corev1.PullPolicy(viper.GetString(constant.KBImagePullPolicy)),
+ SecurityContext: &corev1.SecurityContext{
+ AllowPrivilegeEscalation: boolptr.False(),
+ RunAsUser: &runAsUser,
+ },
+ }
+
+ if r.BackupMethod.RuntimeSettings != nil {
+ container.Resources = r.BackupMethod.RuntimeSettings.Resources
+ }
+
+ if r.ActionSet != nil {
+ container.EnvFrom = r.ActionSet.Spec.EnvFrom
+ }
+
+ intctrlutil.InjectZeroResourcesLimitsIfEmpty(&container)
+
+ podSpec := &corev1.PodSpec{
+ Containers: []corev1.Container{container},
+ Volumes: buildVolumes(),
+ ServiceAccountName: r.targetServiceAccountName(),
+ RestartPolicy: corev1.RestartPolicyNever,
+
+ // tolerate all taints
+ Tolerations: []corev1.Toleration{
+ {
+ Operator: corev1.TolerationOpExists,
+ },
+ },
+ }
+
+ if boolptr.IsSetToTrue(job.RunOnTargetPodNode) {
+ podSpec.NodeSelector = map[string]string{
+ corev1.LabelHostname: targetPod.Spec.NodeName,
+ }
+ }
+ return podSpec
+}
+
+// injectSyncProgressContainer injects a container to sync the backup progress.
+func (r *Request) injectSyncProgressContainer(podSpec *corev1.PodSpec,
+ sync *dpv1alpha1.SyncProgress) {
+ if !boolptr.IsSetToTrue(sync.Enabled) {
+ return
+ }
+
+ // build container to sync backup progress that will update the backup status
+ container := podSpec.Containers[0].DeepCopy()
+ container.Name = syncProgressContainerName
+ container.Image = viper.GetString(constant.KBToolsImage)
+ container.ImagePullPolicy = corev1.PullPolicy(viper.GetString(constant.KBImagePullPolicy))
+ container.Resources = corev1.ResourceRequirements{Limits: nil, Requests: nil}
+ intctrlutil.InjectZeroResourcesLimitsIfEmpty(container)
+ container.Command = []string{"sh", "-c"}
+
+ // append some envs
+ checkIntervalSeconds := int32(5)
+ if sync.IntervalSeconds != nil && *sync.IntervalSeconds > 0 {
+ checkIntervalSeconds = *sync.IntervalSeconds
+ }
+ container.Env = append(container.Env,
+ corev1.EnvVar{
+ Name: dptypes.DPBackupInfoFile,
+ Value: buildBackupInfoFilePath(r.Backup, r.BackupPolicy.Spec.PathPrefix),
+ },
+ corev1.EnvVar{
+ Name: dptypes.DPCheckInterval,
+ Value: fmt.Sprintf("%d", checkIntervalSeconds)},
+ )
+
+ args := fmt.Sprintf(`
+set -o errexit;
+set -o nounset;
+while [ ! -f ${%[1]s} ]; do
+ sleep ${%[2]s}
+done
+backupInfo=$(cat ${%[1]s});
+echo backupInfo:${backupInfo};
+eval kubectl -n %[3]s patch backup %[4]s --subresource=status --type=merge --patch '{\"status\":${backupInfo}}';
+`, dptypes.DPBackupInfoFile, dptypes.DPCheckInterval, r.Backup.Namespace, r.Backup.Name)
+
+ container.Args = []string{args}
+ podSpec.Containers = append(podSpec.Containers, *container)
+}
+
+func (r *Request) backupActionSetExists() bool {
+ return r.ActionSet != nil && r.ActionSet.Spec.Backup != nil
+}
+
+func (r *Request) targetServiceAccountName() string {
+ saName := r.BackupPolicy.Spec.Target.ServiceAccountName
+ if len(saName) > 0 {
+ return saName
+ }
+ // service account name is not specified, use the target pod service account
+ targetPod := r.TargetPods[0]
+ return targetPod.Spec.ServiceAccountName
+}
diff --git a/internal/dataprotection/backup/request_test.go b/internal/dataprotection/backup/request_test.go
new file mode 100644
index 00000000000..03c19330a9b
--- /dev/null
+++ b/internal/dataprotection/backup/request_test.go
@@ -0,0 +1,20 @@
+/*
+Copyright (C) 2022-2023 ApeCloud Co., Ltd
+
+This file is part of KubeBlocks project
+
+This program is free software: you can redistribute it and/or modify
+it under the terms of the GNU Affero General Public License as published by
+the Free Software Foundation, either version 3 of the License, or
+(at your option) any later version.
+
+This program is distributed in the hope that it will be useful
+but WITHOUT ANY WARRANTY; without even the implied warranty of
+MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+GNU Affero General Public License for more details.
+
+You should have received a copy of the GNU Affero General Public License
+along with this program. If not, see .
+*/
+
+package backup
diff --git a/internal/dataprotection/backup/scheduler.go b/internal/dataprotection/backup/scheduler.go
new file mode 100644
index 00000000000..3ac0af7f2e3
--- /dev/null
+++ b/internal/dataprotection/backup/scheduler.go
@@ -0,0 +1,369 @@
+/*
+Copyright (C) 2022-2023 ApeCloud Co., Ltd
+
+This file is part of KubeBlocks project
+
+This program is free software: you can redistribute it and/or modify
+it under the terms of the GNU Affero General Public License as published by
+the Free Software Foundation, either version 3 of the License, or
+(at your option) any later version.
+
+This program is distributed in the hope that it will be useful
+but WITHOUT ANY WARRANTY; without even the implied warranty of
+MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+GNU Affero General Public License for more details.
+
+You should have received a copy of the GNU Affero General Public License
+along with this program. If not, see .
+*/
+
+package backup
+
+import (
+ "fmt"
+ "sort"
+
+ batchv1 "k8s.io/api/batch/v1"
+ corev1 "k8s.io/api/core/v1"
+ metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
+ k8sruntime "k8s.io/apimachinery/pkg/runtime"
+ "k8s.io/apimachinery/pkg/util/json"
+ "sigs.k8s.io/controller-runtime/pkg/client"
+ "sigs.k8s.io/controller-runtime/pkg/controller/controllerutil"
+
+ appsv1alpha1 "github.com/apecloud/kubeblocks/apis/apps/v1alpha1"
+ dpv1alpha1 "github.com/apecloud/kubeblocks/apis/dataprotection/v1alpha1"
+ "github.com/apecloud/kubeblocks/internal/constant"
+ intctrlutil "github.com/apecloud/kubeblocks/internal/controllerutil"
+ dperrors "github.com/apecloud/kubeblocks/internal/dataprotection/errors"
+ dptypes "github.com/apecloud/kubeblocks/internal/dataprotection/types"
+ dputils "github.com/apecloud/kubeblocks/internal/dataprotection/utils"
+ "github.com/apecloud/kubeblocks/internal/dataprotection/utils/boolptr"
+ viper "github.com/apecloud/kubeblocks/internal/viperx"
+)
+
+type Scheduler struct {
+ intctrlutil.RequestCtx
+ Client client.Client
+ Scheme *k8sruntime.Scheme
+ BackupSchedule *dpv1alpha1.BackupSchedule
+ BackupPolicy *dpv1alpha1.BackupPolicy
+}
+
+func (s *Scheduler) Schedule() error {
+ if err := s.validate(); err != nil {
+ return err
+ }
+
+ for i := range s.BackupSchedule.Spec.Schedules {
+ if err := s.handleSchedulePolicy(i); err != nil {
+ return err
+ }
+ }
+ return nil
+}
+
+// validate validates the backup schedule.
+func (s *Scheduler) validate() error {
+ methodInBackupPolicy := func(name string) bool {
+ for _, method := range s.BackupPolicy.Spec.BackupMethods {
+ if method.Name == name {
+ return true
+ }
+ }
+ return false
+ }
+
+ for _, sp := range s.BackupSchedule.Spec.Schedules {
+ if methodInBackupPolicy(sp.BackupMethod) {
+ continue
+ }
+ // backup method name is not in backup policy
+ return fmt.Errorf("backup method %s is not in backup policy %s/%s",
+ sp.BackupMethod, s.BackupPolicy.Namespace, s.BackupPolicy.Name)
+ }
+ return nil
+}
+
+func (s *Scheduler) handleSchedulePolicy(index int) error {
+ schedulePolicy := &s.BackupSchedule.Spec.Schedules[index]
+ // TODO(ldm): better to remove this dependency in the future
+ if err := s.reconfigure(schedulePolicy); err != nil {
+ return err
+ }
+
+ // create/delete/patch cronjob workload
+ return s.reconcileCronJob(schedulePolicy)
+}
+
+type backupReconfigureRef struct {
+ Name string `json:"name"`
+ Key string `json:"key"`
+ Enable parameterPairs `json:"enable,omitempty"`
+ Disable parameterPairs `json:"disable,omitempty"`
+}
+
+type parameterPairs map[string][]appsv1alpha1.ParameterPair
+
+func (s *Scheduler) reconfigure(schedulePolicy *dpv1alpha1.SchedulePolicy) error {
+ reCfgRef := s.BackupSchedule.Annotations[dptypes.ReconfigureRefAnnotationKey]
+ if reCfgRef == "" {
+ return nil
+ }
+ configRef := backupReconfigureRef{}
+ if err := json.Unmarshal([]byte(reCfgRef), &configRef); err != nil {
+ return err
+ }
+
+ enable := boolptr.IsSetToTrue(schedulePolicy.Enabled)
+ if s.BackupSchedule.Annotations[constant.LastAppliedConfigAnnotationKey] == "" && !enable {
+ // disable in the first policy created, no need reconfigure because default configs had been set.
+ return nil
+ }
+ configParameters := configRef.Disable
+ if enable {
+ configParameters = configRef.Enable
+ }
+ if configParameters == nil {
+ return nil
+ }
+ parameters := configParameters[schedulePolicy.BackupMethod]
+ if len(parameters) == 0 {
+ // skip reconfigure if not found parameters.
+ return nil
+ }
+ updateParameterPairsBytes, _ := json.Marshal(parameters)
+ updateParameterPairs := string(updateParameterPairsBytes)
+ if updateParameterPairs == s.BackupSchedule.Annotations[constant.LastAppliedConfigAnnotationKey] {
+ // reconcile the config job if finished
+ return s.reconcileReconfigure()
+ }
+
+ targetPodSelector := s.BackupPolicy.Spec.Target.PodSelector
+ ops := appsv1alpha1.OpsRequest{
+ ObjectMeta: metav1.ObjectMeta{
+ GenerateName: s.BackupSchedule.Name + "-",
+ Namespace: s.BackupSchedule.Namespace,
+ Labels: map[string]string{
+ dptypes.DataProtectionLabelBackupScheduleKey: s.BackupSchedule.Name,
+ },
+ },
+ Spec: appsv1alpha1.OpsRequestSpec{
+ Type: appsv1alpha1.ReconfiguringType,
+ ClusterRef: targetPodSelector.MatchLabels[constant.AppInstanceLabelKey],
+ Reconfigure: &appsv1alpha1.Reconfigure{
+ ComponentOps: appsv1alpha1.ComponentOps{
+ ComponentName: targetPodSelector.MatchLabels[constant.KBAppComponentLabelKey],
+ },
+ Configurations: []appsv1alpha1.ConfigurationItem{
+ {
+ Name: configRef.Name,
+ Keys: []appsv1alpha1.ParameterConfig{
+ {
+ Key: configRef.Key,
+ Parameters: parameters,
+ },
+ },
+ },
+ },
+ },
+ },
+ }
+ if err := s.Client.Create(s.Ctx, &ops); err != nil {
+ return err
+ }
+ s.Recorder.Eventf(s.BackupSchedule, corev1.EventTypeNormal, "Reconfiguring", "update config %s", updateParameterPairs)
+ patch := client.MergeFrom(s.BackupSchedule.DeepCopy())
+ if s.BackupSchedule.Annotations == nil {
+ s.BackupSchedule.Annotations = map[string]string{}
+ }
+ s.BackupSchedule.Annotations[constant.LastAppliedConfigAnnotationKey] = updateParameterPairs
+ if err := s.Client.Patch(s.Ctx, s.BackupSchedule, patch); err != nil {
+ return err
+ }
+ return intctrlutil.NewErrorf(intctrlutil.ErrorTypeRequeue, "requeue to waiting for ops %s finished.", ops.Name)
+}
+
+func (s *Scheduler) reconcileReconfigure() error {
+ opsList := appsv1alpha1.OpsRequestList{}
+ if err := s.Client.List(s.Ctx, &opsList,
+ client.InNamespace(s.BackupSchedule.Namespace),
+ client.MatchingLabels{dptypes.DataProtectionLabelBackupScheduleKey: s.BackupPolicy.Name}); err != nil {
+ return err
+ }
+ if len(opsList.Items) > 0 {
+ sort.Slice(opsList.Items, func(i, j int) bool {
+ return opsList.Items[j].CreationTimestamp.Before(&opsList.Items[i].CreationTimestamp)
+ })
+ latestOps := opsList.Items[0]
+ if latestOps.Status.Phase == appsv1alpha1.OpsFailedPhase {
+ return intctrlutil.NewErrorf(dperrors.ErrorTypeReconfigureFailed, "ops failed %s", latestOps.Name)
+ } else if latestOps.Status.Phase != appsv1alpha1.OpsSucceedPhase {
+ return intctrlutil.NewErrorf(intctrlutil.ErrorTypeRequeue, "waiting for ops %s finished.", latestOps.Name)
+ }
+ }
+ return nil
+}
+
+// buildCronJob builds cronjob from backup schedule.
+func (s *Scheduler) buildCronJob(
+ schedulePolicy *dpv1alpha1.SchedulePolicy,
+ cronJobName string) (*batchv1.CronJob, error) {
+ var (
+ successfulJobsHistoryLimit int32 = 0
+ failedJobsHistoryLimit int32 = 1
+ )
+
+ if cronJobName == "" {
+ cronJobName = GenerateCRNameByBackupSchedule(s.BackupSchedule, schedulePolicy.BackupMethod)
+ }
+
+ podSpec, err := s.buildPodSpec(schedulePolicy)
+ if err != nil {
+ return nil, err
+ }
+
+ cronjob := &batchv1.CronJob{
+ ObjectMeta: metav1.ObjectMeta{
+ Name: cronJobName,
+ Namespace: s.BackupSchedule.Namespace,
+ Labels: map[string]string{
+ constant.AppManagedByLabelKey: constant.AppName,
+ },
+ },
+ Spec: batchv1.CronJobSpec{
+ Schedule: schedulePolicy.CronExpression,
+ SuccessfulJobsHistoryLimit: &successfulJobsHistoryLimit,
+ FailedJobsHistoryLimit: &failedJobsHistoryLimit,
+ ConcurrencyPolicy: batchv1.ForbidConcurrent,
+ JobTemplate: batchv1.JobTemplateSpec{
+ Spec: batchv1.JobSpec{
+ BackoffLimit: s.BackupPolicy.Spec.BackoffLimit,
+ Template: corev1.PodTemplateSpec{
+ Spec: *podSpec,
+ },
+ },
+ },
+ },
+ }
+
+ controllerutil.AddFinalizer(cronjob, dptypes.DataProtectionFinalizerName)
+ // set labels
+ for k, v := range s.BackupSchedule.Labels {
+ if cronjob.Labels == nil {
+ cronjob.SetLabels(map[string]string{})
+ }
+ cronjob.Labels[k] = v
+ }
+ cronjob.Labels[dptypes.DataProtectionLabelBackupScheduleKey] = s.BackupSchedule.Name
+ cronjob.Labels[dptypes.DataProtectionLabelBackupMethodKey] = schedulePolicy.BackupMethod
+ return cronjob, nil
+}
+
+func (s *Scheduler) buildPodSpec(schedulePolicy *dpv1alpha1.SchedulePolicy) (*corev1.PodSpec, error) {
+ // TODO(ldm): add backup deletionPolicy
+ createBackupCmd := fmt.Sprintf(`
+kubectl create -f - < 0 {
+ cronJob = &cronJobList.Items[0]
+ }
+
+ // schedule is disabled, delete cronjob if exists
+ if schedulePolicy == nil || !boolptr.IsSetToTrue(schedulePolicy.Enabled) {
+ if len(cronJob.Name) != 0 {
+ // delete the old cronjob.
+ if err := dputils.RemoveDataProtectionFinalizer(s.Ctx, s.Client, cronJob); err != nil {
+ return err
+ }
+ return s.Client.Delete(s.Ctx, cronJob)
+ }
+ // if no cron expression, return
+ return nil
+ }
+
+ cronjobProto, err := s.buildCronJob(schedulePolicy, cronJob.Name)
+ if err != nil {
+ return err
+ }
+
+ if s.BackupSchedule.Spec.StartingDeadlineMinutes != nil {
+ startingDeadlineSeconds := *s.BackupSchedule.Spec.StartingDeadlineMinutes * 60
+ cronjobProto.Spec.StartingDeadlineSeconds = &startingDeadlineSeconds
+ }
+
+ if len(cronJob.Name) == 0 {
+ // if no cronjob, create it.
+ return s.Client.Create(s.Ctx, cronjobProto)
+ }
+
+ // sync the cronjob with the current backup policy configuration.
+ patch := client.MergeFrom(cronJob.DeepCopy())
+ cronJob.Spec.StartingDeadlineSeconds = cronjobProto.Spec.StartingDeadlineSeconds
+ cronJob.Spec.JobTemplate.Spec.BackoffLimit = s.BackupPolicy.Spec.BackoffLimit
+ cronJob.Spec.JobTemplate.Spec.Template = cronjobProto.Spec.JobTemplate.Spec.Template
+ cronJob.Spec.Schedule = schedulePolicy.CronExpression
+ return s.Client.Patch(s.Ctx, cronJob, patch)
+}
+
+func (s *Scheduler) generateBackupName() string {
+ target := s.BackupPolicy.Spec.Target
+
+ // if cluster name can be found in target labels, use it as backup name prefix
+ backupNamePrefix := target.PodSelector.MatchLabels[constant.AppInstanceLabelKey]
+
+ // if cluster name can not be found, use backup schedule name as backup name prefix
+ if backupNamePrefix == "" {
+ backupNamePrefix = s.BackupSchedule.Name
+ }
+ return backupNamePrefix + "-$(date -u +'%Y%m%d%H%M%S')"
+}
diff --git a/internal/dataprotection/backup/scheduler_test.go b/internal/dataprotection/backup/scheduler_test.go
new file mode 100644
index 00000000000..03c19330a9b
--- /dev/null
+++ b/internal/dataprotection/backup/scheduler_test.go
@@ -0,0 +1,20 @@
+/*
+Copyright (C) 2022-2023 ApeCloud Co., Ltd
+
+This file is part of KubeBlocks project
+
+This program is free software: you can redistribute it and/or modify
+it under the terms of the GNU Affero General Public License as published by
+the Free Software Foundation, either version 3 of the License, or
+(at your option) any later version.
+
+This program is distributed in the hope that it will be useful
+but WITHOUT ANY WARRANTY; without even the implied warranty of
+MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+GNU Affero General Public License for more details.
+
+You should have received a copy of the GNU Affero General Public License
+along with this program. If not, see .
+*/
+
+package backup
diff --git a/internal/dataprotection/backup/suite_test.go b/internal/dataprotection/backup/suite_test.go
new file mode 100644
index 00000000000..455076843c8
--- /dev/null
+++ b/internal/dataprotection/backup/suite_test.go
@@ -0,0 +1,141 @@
+/*
+Copyright (C) 2022-2023 ApeCloud Co., Ltd
+
+This file is part of KubeBlocks project
+
+This program is free software: you can redistribute it and/or modify
+it under the terms of the GNU Affero General Public License as published by
+the Free Software Foundation, either version 3 of the License, or
+(at your option) any later version.
+
+This program is distributed in the hope that it will be useful
+but WITHOUT ANY WARRANTY; without even the implied warranty of
+MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+GNU Affero General Public License for more details.
+
+You should have received a copy of the GNU Affero General Public License
+along with this program. If not, see .
+*/
+
+package backup
+
+import (
+ "context"
+ "go/build"
+ "path/filepath"
+ "testing"
+
+ . "github.com/onsi/ginkgo/v2"
+ . "github.com/onsi/gomega"
+
+ "github.com/go-logr/logr"
+ vsv1 "github.com/kubernetes-csi/external-snapshotter/client/v6/apis/volumesnapshot/v1"
+ "go.uber.org/zap/zapcore"
+ "k8s.io/client-go/kubernetes/scheme"
+ "k8s.io/client-go/rest"
+ "k8s.io/client-go/tools/record"
+ ctrl "sigs.k8s.io/controller-runtime"
+ "sigs.k8s.io/controller-runtime/pkg/client"
+ "sigs.k8s.io/controller-runtime/pkg/envtest"
+ logf "sigs.k8s.io/controller-runtime/pkg/log"
+ "sigs.k8s.io/controller-runtime/pkg/log/zap"
+
+ appsv1alpha1 "github.com/apecloud/kubeblocks/apis/apps/v1alpha1"
+ dpv1alpha1 "github.com/apecloud/kubeblocks/apis/dataprotection/v1alpha1"
+ ctrlutil "github.com/apecloud/kubeblocks/internal/controllerutil"
+ "github.com/apecloud/kubeblocks/internal/testutil"
+ viper "github.com/apecloud/kubeblocks/internal/viperx"
+)
+
+// These tests use Ginkgo (BDD-style Go testing framework). Refer to
+// http://onsi.github.io/ginkgo/ to learn more about Ginkgo.
+
+var (
+ cfg *rest.Config
+ k8sClient client.Client
+ testEnv *envtest.Environment
+ ctx context.Context
+ cancel context.CancelFunc
+ testCtx testutil.TestContext
+ logger logr.Logger
+ recorder record.EventRecorder
+)
+
+func init() {
+ viper.AutomaticEnv()
+}
+
+func TestAction(t *testing.T) {
+ RegisterFailHandler(Fail)
+
+ RunSpecs(t, "Data Protection Backup Suite")
+}
+
+var _ = BeforeSuite(func() {
+ if viper.GetBool("ENABLE_DEBUG_LOG") {
+ logf.SetLogger(zap.New(zap.WriteTo(GinkgoWriter), zap.UseDevMode(true), func(o *zap.Options) {
+ o.TimeEncoder = zapcore.ISO8601TimeEncoder
+ }))
+ }
+
+ ctx, cancel = context.WithCancel(context.TODO())
+ logger = logf.FromContext(ctx).WithValues()
+ logger.Info("logger start")
+
+ By("bootstrapping test environment")
+ testEnv = &envtest.Environment{
+ CRDDirectoryPaths: []string{
+ filepath.Join("..", "..", "..", "config", "crd", "bases"),
+ // use dependent external crds.
+ // resolved by ref: https://github.com/operator-framework/operator-sdk/issues/4434#issuecomment-786794418
+ filepath.Join(build.Default.GOPATH, "pkg", "mod", "github.com", "kubernetes-csi/external-snapshotter/",
+ "client/v6@v6.2.0", "config", "crd"),
+ },
+ ErrorIfCRDPathMissing: true,
+ }
+
+ var err error
+ // cfg is defined in this file globally.
+ cfg, err = testEnv.Start()
+ Expect(err).NotTo(HaveOccurred())
+ Expect(cfg).NotTo(BeNil())
+
+ err = appsv1alpha1.AddToScheme(scheme.Scheme)
+ Expect(err).NotTo(HaveOccurred())
+
+ err = vsv1.AddToScheme(scheme.Scheme)
+ Expect(err).NotTo(HaveOccurred())
+
+ err = dpv1alpha1.AddToScheme(scheme.Scheme)
+ Expect(err).NotTo(HaveOccurred())
+
+ // +kubebuilder:scaffold:scheme
+
+ k8sClient, err = client.New(cfg, client.Options{Scheme: scheme.Scheme})
+ Expect(err).NotTo(HaveOccurred())
+ Expect(k8sClient).NotTo(BeNil())
+
+ // run reconcile
+ k8sManager, err := ctrl.NewManager(cfg, ctrl.Options{
+ Scheme: scheme.Scheme,
+ MetricsBindAddress: "0",
+ ClientDisableCacheFor: ctrlutil.GetUncachedObjects(),
+ })
+ Expect(err).ToNot(HaveOccurred())
+
+ testCtx = testutil.NewDefaultTestContext(ctx, k8sClient, testEnv)
+ recorder = k8sManager.GetEventRecorderFor("dataprotection-backup-test")
+
+ go func() {
+ defer GinkgoRecover()
+ err = k8sManager.Start(ctx)
+ Expect(err).ToNot(HaveOccurred(), "failed to run manager")
+ }()
+})
+
+var _ = AfterSuite(func() {
+ cancel()
+ By("tearing down the test environment")
+ err := testEnv.Stop()
+ Expect(err).NotTo(HaveOccurred())
+})
diff --git a/internal/dataprotection/backup/types.go b/internal/dataprotection/backup/types.go
new file mode 100644
index 00000000000..412cc49ee58
--- /dev/null
+++ b/internal/dataprotection/backup/types.go
@@ -0,0 +1,31 @@
+/*
+Copyright (C) 2022-2023 ApeCloud Co., Ltd
+
+This file is part of KubeBlocks project
+
+This program is free software: you can redistribute it and/or modify
+it under the terms of the GNU Affero General Public License as published by
+the Free Software Foundation, either version 3 of the License, or
+(at your option) any later version.
+
+This program is distributed in the hope that it will be useful
+but WITHOUT ANY WARRANTY; without even the implied warranty of
+MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+GNU Affero General Public License for more details.
+
+You should have received a copy of the GNU Affero General Public License
+along with this program. If not, see .
+*/
+
+package backup
+
+// FormatVersion is the backup file format version, including major, minor, and patch version.
+const (
+ FormatVersion = "0.1.0"
+
+ // RepoVolumeMountPath is the backup repo volume mount path.
+ RepoVolumeMountPath = "/backupdata"
+
+ // backupInfoFileName is the backup info file name in the backup path.
+ backupInfoFileName = "backup.info"
+)
diff --git a/internal/dataprotection/backup/utils.go b/internal/dataprotection/backup/utils.go
new file mode 100644
index 00000000000..95cf776bd90
--- /dev/null
+++ b/internal/dataprotection/backup/utils.go
@@ -0,0 +1,235 @@
+/*
+Copyright (C) 2022-2023 ApeCloud Co., Ltd
+
+This file is part of KubeBlocks project
+
+This program is free software: you can redistribute it and/or modify
+it under the terms of the GNU Affero General Public License as published by
+the Free Software Foundation, either version 3 of the License, or
+(at your option) any later version.
+
+This program is distributed in the hope that it will be useful
+but WITHOUT ANY WARRANTY; without even the implied warranty of
+MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+GNU Affero General Public License for more details.
+
+You should have received a copy of the GNU Affero General Public License
+along with this program. If not, see .
+*/
+
+package backup
+
+import (
+ "context"
+ "fmt"
+ "strings"
+
+ corev1 "k8s.io/api/core/v1"
+ metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
+ "sigs.k8s.io/controller-runtime/pkg/client"
+
+ dpv1alpha1 "github.com/apecloud/kubeblocks/apis/dataprotection/v1alpha1"
+ "github.com/apecloud/kubeblocks/internal/constant"
+ "github.com/apecloud/kubeblocks/internal/dataprotection/action"
+ "github.com/apecloud/kubeblocks/internal/dataprotection/types"
+)
+
+// GetBackupPolicy returns the BackupPolicy with the given namespace and name.
+func GetBackupPolicy(ctx context.Context, cli client.Client, namespace, name string) (*dpv1alpha1.BackupPolicy, error) {
+ backupPolicy := &dpv1alpha1.BackupPolicy{}
+ if err := cli.Get(ctx, client.ObjectKey{Namespace: namespace, Name: name}, backupPolicy); err != nil {
+ return nil, err
+ }
+ return backupPolicy, nil
+}
+
+func GetActionSet(ctx context.Context, cli client.Client, namespace, name string) (*dpv1alpha1.ActionSet, error) {
+ actionSet := &dpv1alpha1.ActionSet{}
+ if err := cli.Get(ctx, client.ObjectKey{Namespace: namespace, Name: name}, actionSet); err != nil {
+ return nil, err
+ }
+ return actionSet, nil
+}
+
+func getVolumesByNames(pod *corev1.Pod, volumeNames []string) []corev1.Volume {
+ var volumes []corev1.Volume
+ for _, v := range pod.Spec.Volumes {
+ for _, name := range volumeNames {
+ if v.Name == name {
+ volumes = append(volumes, v)
+ }
+ }
+ }
+ return volumes
+}
+
+func getVolumesByMounts(pod *corev1.Pod, mounts []corev1.VolumeMount) []corev1.Volume {
+ var volumes []corev1.Volume
+ for _, v := range pod.Spec.Volumes {
+ for _, m := range mounts {
+ if v.Name == m.Name {
+ volumes = append(volumes, v)
+ }
+ }
+ }
+ return volumes
+}
+
+// TODO: if the result is empty, should we return the pod's volumes?
+//
+// if volumes can not found in the pod spec, maybe output a warning log?
+func getVolumesByVolumeInfo(pod *corev1.Pod, volumeInfo *dpv1alpha1.TargetVolumeInfo) []corev1.Volume {
+ if volumeInfo == nil {
+ return nil
+ }
+ var volumes []corev1.Volume
+ if len(volumeInfo.Volumes) > 0 {
+ volumes = getVolumesByNames(pod, volumeInfo.Volumes)
+ } else if len(volumeInfo.VolumeMounts) > 0 {
+ volumes = getVolumesByMounts(pod, volumeInfo.VolumeMounts)
+ }
+ return volumes
+}
+
+func getVolumeMountsByVolumeInfo(pod *corev1.Pod, info *dpv1alpha1.TargetVolumeInfo) []corev1.VolumeMount {
+ if info == nil || len(info.VolumeMounts) == 0 {
+ return nil
+ }
+ var mounts []corev1.VolumeMount
+ for _, v := range pod.Spec.Volumes {
+ for _, m := range info.VolumeMounts {
+ if v.Name == m.Name {
+ mounts = append(mounts, m)
+ }
+ }
+ }
+ return mounts
+}
+
+func getPVCsByVolumeNames(cli client.Client,
+ pod *corev1.Pod,
+ volumeNames []string) ([]action.PersistentVolumeClaimWrapper, error) {
+ if len(volumeNames) == 0 {
+ return nil, nil
+ }
+ var all []action.PersistentVolumeClaimWrapper
+ for _, v := range pod.Spec.Volumes {
+ if v.PersistentVolumeClaim == nil {
+ continue
+ }
+ for _, name := range volumeNames {
+ if v.Name != name {
+ continue
+ }
+ // get the PVC from pod's volumes
+ tmp := corev1.PersistentVolumeClaim{}
+ pvcKey := client.ObjectKey{Namespace: pod.Namespace, Name: v.PersistentVolumeClaim.ClaimName}
+ if err := cli.Get(context.Background(), pvcKey, &tmp); err != nil {
+ return nil, err
+ }
+
+ all = append(all, action.NewPersistentVolumeClaimWrapper(*tmp.DeepCopy(), name))
+ }
+ }
+ return all, nil
+}
+
+func GenerateBackupRepoVolumeName(pvcName string) string {
+ return fmt.Sprintf("dp-backup-%s", pvcName)
+}
+
+func buildBackupRepoVolume(pvcName string) corev1.Volume {
+ return corev1.Volume{
+ Name: GenerateBackupRepoVolumeName(pvcName),
+ VolumeSource: corev1.VolumeSource{
+ PersistentVolumeClaim: &corev1.PersistentVolumeClaimVolumeSource{
+ ClaimName: pvcName,
+ },
+ },
+ }
+}
+
+func buildBackupRepoVolumeMount(pvcName string) corev1.VolumeMount {
+ return corev1.VolumeMount{
+ Name: GenerateBackupRepoVolumeName(pvcName),
+ MountPath: RepoVolumeMountPath,
+ }
+}
+
+func excludeLabelsForWorkload() []string {
+ return []string{constant.KBAppComponentLabelKey}
+}
+
+// BuildBackupWorkloadLabels builds the labels for workload which owned by backup.
+func BuildBackupWorkloadLabels(backup *dpv1alpha1.Backup) map[string]string {
+ labels := backup.Labels
+ if labels == nil {
+ labels = map[string]string{}
+ } else {
+ for _, v := range excludeLabelsForWorkload() {
+ delete(labels, v)
+ }
+ }
+ labels[types.DataProtectionLabelBackupNameKey] = backup.Name
+ return labels
+}
+
+func buildBackupJobObjMeta(backup *dpv1alpha1.Backup, prefix string) *metav1.ObjectMeta {
+ return &metav1.ObjectMeta{
+ Name: GenerateBackupJobName(backup, prefix),
+ Namespace: backup.Namespace,
+ Labels: BuildBackupWorkloadLabels(backup),
+ }
+}
+
+func GenerateBackupJobName(backup *dpv1alpha1.Backup, prefix string) string {
+ name := fmt.Sprintf("%s-%s-%s", prefix, backup.Name, backup.UID[:8])
+ // job name cannot exceed 63 characters for label name limit.
+ if len(name) > 63 {
+ return name[:63]
+ }
+ return name
+}
+
+// GenerateCRNameByBackupSchedule generate a CR name which is created by BackupSchedule, such as CronJob Backup.
+func GenerateCRNameByBackupSchedule(backupSchedule *dpv1alpha1.BackupSchedule, method string) string {
+ name := fmt.Sprintf("%s-%s", generateUniqueNameWithBackupSchedule(backupSchedule), backupSchedule.Namespace)
+ if len(name) > 30 {
+ name = strings.TrimRight(name[:30], "-")
+ }
+ return fmt.Sprintf("%s-%s", name, method)
+}
+
+func generateUniqueNameWithBackupSchedule(backupSchedule *dpv1alpha1.BackupSchedule) string {
+ uniqueName := backupSchedule.Name
+ if len(backupSchedule.OwnerReferences) > 0 {
+ uniqueName = fmt.Sprintf("%s-%s", backupSchedule.OwnerReferences[0].UID[:8], backupSchedule.OwnerReferences[0].Name)
+ }
+ return uniqueName
+}
+
+func buildBackupInfoFilePath(backup *dpv1alpha1.Backup, pathPrefix string) string {
+ return buildBackupPathInContainer(backup, pathPrefix) + "/" + backupInfoFileName
+}
+
+func buildBackupPathInContainer(backup *dpv1alpha1.Backup, pathPrefix string) string {
+ return RepoVolumeMountPath + BuildBackupPath(backup, pathPrefix)
+}
+
+// BuildBackupPath builds the path to storage backup datas in backup repository.
+func BuildBackupPath(backup *dpv1alpha1.Backup, pathPrefix string) string {
+ pathPrefix = strings.TrimRight(pathPrefix, "/")
+ if strings.TrimSpace(pathPrefix) == "" || strings.HasPrefix(pathPrefix, "/") {
+ return fmt.Sprintf("/%s%s/%s", backup.Namespace, pathPrefix, backup.Name)
+ }
+ return fmt.Sprintf("/%s/%s/%s", backup.Namespace, pathPrefix, backup.Name)
+}
+
+func GetSchedulePolicyByMethod(backupSchedule *dpv1alpha1.BackupSchedule, method string) *dpv1alpha1.SchedulePolicy {
+ for _, s := range backupSchedule.Spec.Schedules {
+ if s.BackupMethod == method {
+ return &s
+ }
+ }
+ return nil
+}
diff --git a/internal/dataprotection/builder/builder.go b/internal/dataprotection/builder/builder.go
new file mode 100644
index 00000000000..cbbe980f497
--- /dev/null
+++ b/internal/dataprotection/builder/builder.go
@@ -0,0 +1,41 @@
+/*
+Copyright (C) 2022-2023 ApeCloud Co., Ltd
+
+This file is part of KubeBlocks project
+
+This program is free software: you can redistribute it and/or modify
+it under the terms of the GNU Affero General Public License as published by
+the Free Software Foundation, either version 3 of the License, or
+(at your option) any later version.
+
+This program is distributed in the hope that it will be useful
+but WITHOUT ANY WARRANTY; without even the implied warranty of
+MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+GNU Affero General Public License for more details.
+
+You should have received a copy of the GNU Affero General Public License
+along with this program. If not, see .
+*/
+
+package builder
+
+import (
+ vsv1 "github.com/kubernetes-csi/external-snapshotter/client/v6/apis/volumesnapshot/v1"
+ metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
+
+ "github.com/apecloud/kubeblocks/internal/constant"
+)
+
+func BuildVolumeSnapshotClass(name string, driver string) *vsv1.VolumeSnapshotClass {
+ vsc := &vsv1.VolumeSnapshotClass{
+ ObjectMeta: metav1.ObjectMeta{
+ Name: name,
+ Labels: map[string]string{
+ constant.KBManagedByKey: constant.AppName,
+ },
+ },
+ Driver: driver,
+ DeletionPolicy: vsv1.VolumeSnapshotContentDelete,
+ }
+ return vsc
+}
diff --git a/internal/dataprotection/builder/builder_test.go b/internal/dataprotection/builder/builder_test.go
new file mode 100644
index 00000000000..b10522e9964
--- /dev/null
+++ b/internal/dataprotection/builder/builder_test.go
@@ -0,0 +1,36 @@
+/*
+Copyright (C) 2022-2023 ApeCloud Co., Ltd
+
+This file is part of KubeBlocks project
+
+This program is free software: you can redistribute it and/or modify
+it under the terms of the GNU Affero General Public License as published by
+the Free Software Foundation, either version 3 of the License, or
+(at your option) any later version.
+
+This program is distributed in the hope that it will be useful
+but WITHOUT ANY WARRANTY; without even the implied warranty of
+MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+GNU Affero General Public License for more details.
+
+You should have received a copy of the GNU Affero General Public License
+along with this program. If not, see .
+*/
+
+package builder
+
+import (
+ . "github.com/onsi/ginkgo/v2"
+ . "github.com/onsi/gomega"
+)
+
+var _ = Describe("builder", func() {
+ It("builds volume snapshot class correctly", func() {
+ className := "vsc-test"
+ driverName := "csi-driver-test"
+ obj := BuildVolumeSnapshotClass(className, driverName)
+ Expect(obj).ShouldNot(BeNil())
+ Expect(obj.Name).Should(Equal(className))
+ Expect(obj.Driver).Should(Equal(driverName))
+ })
+})
diff --git a/internal/dataprotection/builder/suite_test.go b/internal/dataprotection/builder/suite_test.go
new file mode 100644
index 00000000000..7cb19699de3
--- /dev/null
+++ b/internal/dataprotection/builder/suite_test.go
@@ -0,0 +1,47 @@
+/*
+Copyright (C) 2022-2023 ApeCloud Co., Ltd
+
+This file is part of KubeBlocks project
+
+This program is free software: you can redistribute it and/or modify
+it under the terms of the GNU Affero General Public License as published by
+the Free Software Foundation, either version 3 of the License, or
+(at your option) any later version.
+
+This program is distributed in the hope that it will be useful
+but WITHOUT ANY WARRANTY; without even the implied warranty of
+MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+GNU Affero General Public License for more details.
+
+You should have received a copy of the GNU Affero General Public License
+along with this program. If not, see .
+*/
+
+package builder
+
+import (
+ "testing"
+
+ . "github.com/onsi/ginkgo/v2"
+ . "github.com/onsi/gomega"
+)
+
+// These tests use Ginkgo (BDD-style Go testing framework). Refer to
+// http://onsi.github.io/ginkgo/ to learn more about Ginkgo.
+
+func TestAPIs(t *testing.T) {
+ RegisterFailHandler(Fail)
+
+ RunSpecs(t, "Data Protection Builder Suite")
+}
+
+var _ = BeforeSuite(func() {
+ // +kubebuilder:scaffold:scheme
+
+ go func() {
+ defer GinkgoRecover()
+ }()
+})
+
+var _ = AfterSuite(func() {
+})
diff --git a/internal/dataprotection/errors/errors.go b/internal/dataprotection/errors/errors.go
new file mode 100644
index 00000000000..207f374bd32
--- /dev/null
+++ b/internal/dataprotection/errors/errors.go
@@ -0,0 +1,85 @@
+/*
+Copyright (C) 2022-2023 ApeCloud Co., Ltd
+
+This file is part of KubeBlocks project
+
+This program is free software: you can redistribute it and/or modify
+it under the terms of the GNU Affero General Public License as published by
+the Free Software Foundation, either version 3 of the License, or
+(at your option) any later version.
+
+This program is distributed in the hope that it will be useful
+but WITHOUT ANY WARRANTY; without even the implied warranty of
+MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+GNU Affero General Public License for more details.
+
+You should have received a copy of the GNU Affero General Public License
+along with this program. If not, see .
+*/
+
+package errors
+
+import (
+ intctrlutil "github.com/apecloud/kubeblocks/internal/controllerutil"
+)
+
+// ErrorType for backup
+const (
+ // ErrorTypeBackupNotSupported this backup type not supported
+ ErrorTypeBackupNotSupported intctrlutil.ErrorType = "BackupNotSupported"
+ // ErrorTypeBackupPVTemplateNotFound this pv template not found
+ ErrorTypeBackupPVTemplateNotFound intctrlutil.ErrorType = "BackupPVTemplateNotFound"
+ // ErrorTypeBackupNotCompleted report backup not completed.
+ ErrorTypeBackupNotCompleted intctrlutil.ErrorType = "BackupNotCompleted"
+ // ErrorTypeBackupPVCNameIsEmpty pvc name for backup is empty
+ ErrorTypeBackupPVCNameIsEmpty intctrlutil.ErrorType = "BackupPVCNameIsEmpty"
+ // ErrorTypeBackupJobFailed backup job failed
+ ErrorTypeBackupJobFailed intctrlutil.ErrorType = "BackupJobFailed"
+ // ErrorTypeStorageNotMatch storage not match
+ ErrorTypeStorageNotMatch intctrlutil.ErrorType = "ErrorTypeStorageNotMatch"
+ // ErrorTypeReconfigureFailed reconfigure failed
+ ErrorTypeReconfigureFailed intctrlutil.ErrorType = "ErrorTypeReconfigureFailed"
+ // ErrorTypeInvalidLogfileBackupName invalid logfile backup name
+ ErrorTypeInvalidLogfileBackupName intctrlutil.ErrorType = "InvalidLogfileBackupName"
+ // ErrorTypeBackupScheduleDisabled backup schedule disabled
+ ErrorTypeBackupScheduleDisabled intctrlutil.ErrorType = "BackupScheduleDisabled"
+ // ErrorTypeLogfileScheduleDisabled logfile schedule disabled
+ ErrorTypeLogfileScheduleDisabled intctrlutil.ErrorType = "LogfileScheduleDisabled"
+ // ErrorTypeWaitForExternalHandler wait for external handler to handle the Backup or Restore
+ ErrorTypeWaitForExternalHandler intctrlutil.ErrorType = "WaitForExternalHandler"
+)
+
+// NewBackupNotSupported returns a new Error with ErrorTypeBackupNotSupported.
+func NewBackupNotSupported(backupType, backupPolicyName string) *intctrlutil.Error {
+ return intctrlutil.NewErrorf(ErrorTypeBackupNotSupported, `backup type "%s" not supported by backup policy "%s"`, backupType, backupPolicyName)
+}
+
+// NewBackupPVTemplateNotFound returns a new Error with ErrorTypeBackupPVTemplateNotFound.
+func NewBackupPVTemplateNotFound(cmName, cmNamespace string) *intctrlutil.Error {
+ return intctrlutil.NewErrorf(ErrorTypeBackupPVTemplateNotFound, `"the persistentVolume template is empty in the configMap %s/%s", pvConfig.Namespace, pvConfig.Name`, cmNamespace, cmName)
+}
+
+// NewBackupPVCNameIsEmpty returns a new Error with ErrorTypeBackupPVCNameIsEmpty.
+func NewBackupPVCNameIsEmpty(backupRepo, backupPolicyName string) *intctrlutil.Error {
+ return intctrlutil.NewErrorf(ErrorTypeBackupPVCNameIsEmpty, `the persistentVolumeClaim name of %s is empty in BackupPolicy "%s"`, backupRepo, backupPolicyName)
+}
+
+// NewBackupJobFailed returns a new Error with ErrorTypeBackupJobFailed.
+func NewBackupJobFailed(jobName string) *intctrlutil.Error {
+ return intctrlutil.NewErrorf(ErrorTypeBackupJobFailed, `backup job "%s" failed`, jobName)
+}
+
+// NewInvalidLogfileBackupName returns a new Error with ErrorTypeInvalidLogfileBackupName.
+func NewInvalidLogfileBackupName(backupPolicyName string) *intctrlutil.Error {
+ return intctrlutil.NewErrorf(ErrorTypeInvalidLogfileBackupName, `backup name is incorrect for logfile, you can create the logfile backup by enabling the schedule in BackupPolicy "%s"`, backupPolicyName)
+}
+
+// NewBackupScheduleDisabled returns a new Error with ErrorTypeBackupScheduleDisabled.
+func NewBackupScheduleDisabled(backupType, backupPolicyName string) *intctrlutil.Error {
+ return intctrlutil.NewErrorf(ErrorTypeBackupScheduleDisabled, `%s schedule is disabled, you can enable spec.schedule.%s in BackupPolicy "%s"`, backupType, backupType, backupPolicyName)
+}
+
+// NewBackupLogfileScheduleDisabled returns a new Error with ErrorTypeLogfileScheduleDisabled.
+func NewBackupLogfileScheduleDisabled(backupToolName string) *intctrlutil.Error {
+ return intctrlutil.NewErrorf(ErrorTypeLogfileScheduleDisabled, `BackupTool "%s" of the backup relies on logfile. Please enable the logfile scheduling firstly`, backupToolName)
+}
diff --git a/internal/dataprotection/errors/errors_test.go b/internal/dataprotection/errors/errors_test.go
new file mode 100644
index 00000000000..fde1dcf9335
--- /dev/null
+++ b/internal/dataprotection/errors/errors_test.go
@@ -0,0 +1,229 @@
+/*
+Copyright (C) 2022-2023 ApeCloud Co., Ltd
+
+This file is part of KubeBlocks project
+
+This program is free software: you can redistribute it and/or modify
+it under the terms of the GNU Affero General Public License as published by
+the Free Software Foundation, either version 3 of the License, or
+(at your option) any later version.
+
+This program is distributed in the hope that it will be useful
+but WITHOUT ANY WARRANTY; without even the implied warranty of
+MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+GNU Affero General Public License for more details.
+
+You should have received a copy of the GNU Affero General Public License
+along with this program. If not, see .
+*/
+
+package errors
+
+import (
+ "fmt"
+ "reflect"
+ "testing"
+
+ "github.com/pkg/errors"
+
+ intctrlutil "github.com/apecloud/kubeblocks/internal/controllerutil"
+)
+
+func TestNerError(t *testing.T) {
+ err1 := intctrlutil.NewError(ErrorTypeBackupNotCompleted, "test c2")
+ if err1.Error() != "test c2" {
+ t.Error("NewErrorf failed")
+ }
+}
+
+func TestNerErrorf(t *testing.T) {
+ err1 := intctrlutil.NewErrorf(ErrorTypeBackupNotCompleted, "test %s %s", "c1", "c2")
+ if err1.Error() != "test c1 c2" {
+ t.Error("NewErrorf failed")
+ }
+ testError := fmt.Errorf("test: %w", err1)
+ if !errors.Is(testError, err1) {
+ t.Error("errors.Is failed")
+ }
+
+ var target *intctrlutil.Error
+ if !errors.As(testError, &target) {
+ t.Error("errors.As failed")
+ }
+}
+
+func TestNewErrors(t *testing.T) {
+ backupNotSupported := NewBackupNotSupported("datafile", "policy-test")
+ if !intctrlutil.IsTargetError(backupNotSupported, ErrorTypeBackupNotSupported) {
+ t.Error("should be error of BackupNotSupported")
+ }
+ pvTemplateNotFound := NewBackupPVTemplateNotFound("configName", "default")
+ if !intctrlutil.IsTargetError(pvTemplateNotFound, ErrorTypeBackupPVTemplateNotFound) {
+ t.Error("should be error of BackupPVTemplateNotFound")
+ }
+ pvsIsEmpty := NewBackupPVCNameIsEmpty("datafile", "policy-test1")
+ if !intctrlutil.IsTargetError(pvsIsEmpty, ErrorTypeBackupPVCNameIsEmpty) {
+ t.Error("should be error of BackupPVCNameIsEmpty")
+ }
+ jobFailed := NewBackupJobFailed("jobName")
+ if !intctrlutil.IsTargetError(jobFailed, ErrorTypeBackupJobFailed) {
+ t.Error("should be error of BackupJobFailed")
+ }
+}
+
+func TestUnwrapControllerError(t *testing.T) {
+ backupNotSupported := NewBackupNotSupported("datafile", "policy-test")
+ newErr := intctrlutil.UnwrapControllerError(backupNotSupported)
+ if newErr == nil {
+ t.Error("should unwrap a controller error, but got nil")
+ }
+ err := errors.New("test error")
+ newErr = intctrlutil.UnwrapControllerError(err)
+ if newErr != nil {
+ t.Errorf("should not unwrap a controller error, but got: %v", newErr)
+ }
+}
+
+func TestNewBackupJobFailed(t *testing.T) {
+ type args struct {
+ jobName string
+ }
+ tests := []struct {
+ name string
+ args args
+ want *intctrlutil.Error
+ }{
+ // TODO: Add test cases.
+ }
+ for _, tt := range tests {
+ t.Run(tt.name, func(t *testing.T) {
+ if got := NewBackupJobFailed(tt.args.jobName); !reflect.DeepEqual(got, tt.want) {
+ t.Errorf("NewBackupJobFailed() = %v, want %v", got, tt.want)
+ }
+ })
+ }
+}
+
+func TestNewBackupLogfileScheduleDisabled(t *testing.T) {
+ type args struct {
+ backupToolName string
+ }
+ tests := []struct {
+ name string
+ args args
+ want *intctrlutil.Error
+ }{
+ // TODO: Add test cases.
+ }
+ for _, tt := range tests {
+ t.Run(tt.name, func(t *testing.T) {
+ if got := NewBackupLogfileScheduleDisabled(tt.args.backupToolName); !reflect.DeepEqual(got, tt.want) {
+ t.Errorf("NewBackupLogfileScheduleDisabled() = %v, want %v", got, tt.want)
+ }
+ })
+ }
+}
+
+func TestNewBackupNotSupported(t *testing.T) {
+ type args struct {
+ backupType string
+ backupPolicyName string
+ }
+ tests := []struct {
+ name string
+ args args
+ want *intctrlutil.Error
+ }{
+ // TODO: Add test cases.
+ }
+ for _, tt := range tests {
+ t.Run(tt.name, func(t *testing.T) {
+ if got := NewBackupNotSupported(tt.args.backupType, tt.args.backupPolicyName); !reflect.DeepEqual(got, tt.want) {
+ t.Errorf("NewBackupNotSupported() = %v, want %v", got, tt.want)
+ }
+ })
+ }
+}
+
+func TestNewBackupPVCNameIsEmpty(t *testing.T) {
+ type args struct {
+ backupRepo string
+ backupPolicyName string
+ }
+ tests := []struct {
+ name string
+ args args
+ want *intctrlutil.Error
+ }{
+ // TODO: Add test cases.
+ }
+ for _, tt := range tests {
+ t.Run(tt.name, func(t *testing.T) {
+ if got := NewBackupPVCNameIsEmpty(tt.args.backupRepo, tt.args.backupPolicyName); !reflect.DeepEqual(got, tt.want) {
+ t.Errorf("NewBackupPVCNameIsEmpty() = %v, want %v", got, tt.want)
+ }
+ })
+ }
+}
+
+func TestNewBackupPVTemplateNotFound(t *testing.T) {
+ type args struct {
+ cmName string
+ cmNamespace string
+ }
+ tests := []struct {
+ name string
+ args args
+ want *intctrlutil.Error
+ }{
+ // TODO: Add test cases.
+ }
+ for _, tt := range tests {
+ t.Run(tt.name, func(t *testing.T) {
+ if got := NewBackupPVTemplateNotFound(tt.args.cmName, tt.args.cmNamespace); !reflect.DeepEqual(got, tt.want) {
+ t.Errorf("NewBackupPVTemplateNotFound() = %v, want %v", got, tt.want)
+ }
+ })
+ }
+}
+
+func TestNewBackupScheduleDisabled(t *testing.T) {
+ type args struct {
+ backupType string
+ backupPolicyName string
+ }
+ tests := []struct {
+ name string
+ args args
+ want *intctrlutil.Error
+ }{
+ // TODO: Add test cases.
+ }
+ for _, tt := range tests {
+ t.Run(tt.name, func(t *testing.T) {
+ if got := NewBackupScheduleDisabled(tt.args.backupType, tt.args.backupPolicyName); !reflect.DeepEqual(got, tt.want) {
+ t.Errorf("NewBackupScheduleDisabled() = %v, want %v", got, tt.want)
+ }
+ })
+ }
+}
+
+func TestNewInvalidLogfileBackupName(t *testing.T) {
+ type args struct {
+ backupPolicyName string
+ }
+ tests := []struct {
+ name string
+ args args
+ want *intctrlutil.Error
+ }{
+ // TODO: Add test cases.
+ }
+ for _, tt := range tests {
+ t.Run(tt.name, func(t *testing.T) {
+ if got := NewInvalidLogfileBackupName(tt.args.backupPolicyName); !reflect.DeepEqual(got, tt.want) {
+ t.Errorf("NewInvalidLogfileBackupName() = %v, want %v", got, tt.want)
+ }
+ })
+ }
+}
diff --git a/internal/dataprotection/restore/builder.go b/internal/dataprotection/restore/builder.go
new file mode 100644
index 00000000000..128cbd28274
--- /dev/null
+++ b/internal/dataprotection/restore/builder.go
@@ -0,0 +1,294 @@
+/*
+Copyright (C) 2022-2023 ApeCloud Co., Ltd
+
+This file is part of KubeBlocks project
+
+This program is free software: you can redistribute it and/or modify
+it under the terms of the GNU Affero General Public License as published by
+the Free Software Foundation, either version 3 of the License, or
+(at your option) any later version.
+
+This program is distributed in the hope that it will be useful
+but WITHOUT ANY WARRANTY; without even the implied warranty of
+MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+GNU Affero General Public License for more details.
+
+You should have received a copy of the GNU Affero General Public License
+along with this program. If not, see .
+*/
+
+package restore
+
+import (
+ "fmt"
+ "strconv"
+ "strings"
+ "time"
+
+ batchv1 "k8s.io/api/batch/v1"
+ corev1 "k8s.io/api/core/v1"
+ metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
+ "sigs.k8s.io/controller-runtime/pkg/controller/controllerutil"
+
+ dpv1alpha1 "github.com/apecloud/kubeblocks/apis/dataprotection/v1alpha1"
+ intctrlutil "github.com/apecloud/kubeblocks/internal/controllerutil"
+ dptypes "github.com/apecloud/kubeblocks/internal/dataprotection/types"
+ "github.com/apecloud/kubeblocks/internal/dataprotection/utils"
+)
+
+type restoreJobBuilder struct {
+ restore *dpv1alpha1.Restore
+ stage dpv1alpha1.RestoreStage
+ backupSet BackupActionSet
+ env []corev1.EnvVar
+ commonVolumes []corev1.Volume
+ commonVolumeMounts []corev1.VolumeMount
+ // specificVolumes should be rebuilt for each job.
+ specificVolumes []corev1.Volume
+ // specificVolumeMounts should be rebuilt for each job.
+ specificVolumeMounts []corev1.VolumeMount
+ image string
+ command []string
+ tolerations []corev1.Toleration
+ nodeSelector map[string]string
+}
+
+func newRestoreJobBuilder(restore *dpv1alpha1.Restore, backupSet BackupActionSet, stage dpv1alpha1.RestoreStage) *restoreJobBuilder {
+ return &restoreJobBuilder{
+ restore: restore,
+ backupSet: backupSet,
+ stage: stage,
+ commonVolumes: []corev1.Volume{},
+ commonVolumeMounts: []corev1.VolumeMount{},
+ }
+}
+
+func (r *restoreJobBuilder) buildPVCVolumeAndMount(
+ claim dpv1alpha1.RestoreVolumeClaim,
+ identifier string) (*corev1.Volume, *corev1.VolumeMount, error) {
+ volumeName := fmt.Sprintf("%s-%s", identifier, claim.Name)
+ volume := &corev1.Volume{
+ Name: volumeName,
+ VolumeSource: corev1.VolumeSource{PersistentVolumeClaim: &corev1.PersistentVolumeClaimVolumeSource{ClaimName: claim.Name}},
+ }
+ volumeMount := &corev1.VolumeMount{Name: volumeName}
+ if claim.MountPath != "" {
+ volumeMount.MountPath = claim.MountPath
+ return volume, volumeMount, nil
+ }
+ mountPath := getMountPathWithSourceVolume(r.backupSet.Backup, claim.VolumeSource)
+ if mountPath != "" {
+ volumeMount.MountPath = mountPath
+ return volume, volumeMount, nil
+ }
+
+ if r.backupSet.UseVolumeSnapshot && !r.backupSet.ActionSet.HasPrepareDataStage() {
+ return nil, nil, nil
+ }
+ return nil, nil, intctrlutil.NewFatalError(fmt.Sprintf(`unable to find the mountPath corresponding to volumeSource "%s" from status.backupMethod.targetVolumes.volumeMounts of backup "%s"`,
+ claim.VolumeSource, r.backupSet.Backup.Name))
+}
+
+// addBackupVolumeAndMount adds the volume and volumeMount of backup pvc to common volumes and volumeMounts slice.
+func (r *restoreJobBuilder) addBackupVolumeAndMount() *restoreJobBuilder {
+ if r.backupSet.Backup.Status.PersistentVolumeClaimName != "" {
+ backupName := r.backupSet.Backup.Name
+ r.commonVolumes = append(r.commonVolumes, corev1.Volume{
+ Name: backupName,
+ VolumeSource: corev1.VolumeSource{
+ PersistentVolumeClaim: &corev1.PersistentVolumeClaimVolumeSource{ClaimName: r.backupSet.Backup.Status.PersistentVolumeClaimName},
+ },
+ })
+ r.commonVolumeMounts = append(r.commonVolumeMounts, corev1.VolumeMount{
+ Name: backupName,
+ MountPath: "/" + backupName,
+ })
+ }
+ return r
+}
+
+// addToCommonVolumesAndMounts adds the volume and volumeMount to common volumes and volumeMounts slice.
+func (r *restoreJobBuilder) addToCommonVolumesAndMounts(volume *corev1.Volume, volumeMount *corev1.VolumeMount) *restoreJobBuilder {
+ if volume != nil {
+ r.commonVolumes = append(r.commonVolumes, *volume)
+ }
+ if volumeMount != nil {
+ r.commonVolumeMounts = append(r.commonVolumeMounts, *volumeMount)
+ }
+ return r
+}
+
+// resetSpecificVolumesAndMounts resets the specific volumes and volumeMounts slice.
+func (r *restoreJobBuilder) resetSpecificVolumesAndMounts() {
+ r.specificVolumes = []corev1.Volume{}
+ r.specificVolumeMounts = []corev1.VolumeMount{}
+}
+
+// addToSpecificVolumesAndMounts adds the volume and volumeMount to specific volumes and volumeMounts slice.
+func (r *restoreJobBuilder) addToSpecificVolumesAndMounts(volume *corev1.Volume, volumeMount *corev1.VolumeMount) *restoreJobBuilder {
+ if volume != nil {
+ r.specificVolumes = append(r.specificVolumes, *volume)
+ }
+ if volumeMount != nil {
+ r.specificVolumeMounts = append(r.specificVolumeMounts, *volumeMount)
+ }
+ return r
+}
+
+func (r *restoreJobBuilder) setImage(image string) *restoreJobBuilder {
+ r.image = image
+ return r
+}
+
+func (r *restoreJobBuilder) setCommand(command []string) *restoreJobBuilder {
+ r.command = command
+ return r
+}
+
+func (r *restoreJobBuilder) setToleration(tolerations []corev1.Toleration) *restoreJobBuilder {
+ r.tolerations = tolerations
+ return r
+}
+
+func (r *restoreJobBuilder) setNodeNameToNodeSelector(nodeName string) *restoreJobBuilder {
+ r.nodeSelector = map[string]string{
+ corev1.LabelHostname: nodeName,
+ }
+ return r
+}
+
+// addCommonEnv adds the common envs for each restore job.
+func (r *restoreJobBuilder) addCommonEnv() *restoreJobBuilder {
+ backupName := r.backupSet.Backup.Name
+ // add backupName env
+ r.env = []corev1.EnvVar{{Name: dptypes.DPBackupName, Value: backupName}}
+ // add mount path env of backup dir
+ filePath := r.backupSet.Backup.Status.Path
+ if filePath != "" {
+ r.env = append(r.env, corev1.EnvVar{Name: dptypes.DPBackupDIR, Value: fmt.Sprintf("/%s%s", backupName, filePath)})
+ // TODO: add continuous file path env
+ }
+ // add time env
+ actionSetEnv := r.backupSet.ActionSet.Spec.Env
+ timeFormat := getTimeFormat(r.backupSet.ActionSet.Spec.Env)
+ appendTimeEnv := func(envName, envTimestampName string, targetTime *metav1.Time) {
+ if targetTime.IsZero() {
+ return
+ }
+ if envName != "" {
+ r.env = append(r.env, corev1.EnvVar{Name: envName, Value: targetTime.UTC().Format(timeFormat)})
+ }
+ if envTimestampName != "" {
+ r.env = append(r.env, corev1.EnvVar{Name: envTimestampName, Value: strconv.FormatInt(targetTime.Unix(), 10)})
+ }
+ }
+ appendTimeEnv(dptypes.DPBackupStopTime, "", r.backupSet.Backup.GetEndTime())
+ if r.restore.Spec.RestoreTime != "" {
+ restoreTime, _ := time.Parse(time.RFC3339, r.restore.Spec.RestoreTime)
+ appendTimeEnv(DPRestoreTime, DPRestoreTimestamp, &metav1.Time{Time: restoreTime})
+ }
+ // append actionSet env
+ r.env = append(r.env, actionSetEnv...)
+ backupMethod := r.backupSet.Backup.Status.BackupMethod
+ if backupMethod != nil && len(backupMethod.Env) > 0 {
+ r.env = utils.MergeEnv(r.env, backupMethod.Env)
+ }
+ // merge the restore env
+ r.env = utils.MergeEnv(r.env, r.restore.Spec.Env)
+ return r
+}
+
+func (r *restoreJobBuilder) addTargetPodAndCredentialEnv(pod *corev1.Pod, connectCredential *dpv1alpha1.ConnectCredential) *restoreJobBuilder {
+ if pod == nil {
+ return r
+ }
+ var env []corev1.EnvVar
+ // Note: now only add the first container envs.
+ if len(pod.Spec.Containers) != 0 {
+ env = pod.Spec.Containers[0].Env
+ }
+ env = append(env, corev1.EnvVar{Name: dptypes.DPDBHost, Value: intctrlutil.BuildPodHostDNS(pod)})
+ if connectCredential != nil {
+ appendEnvFromSecret := func(envName, keyName string) {
+ if keyName != "" {
+ env = append(env, corev1.EnvVar{Name: envName, ValueFrom: &corev1.EnvVarSource{
+ SecretKeyRef: &corev1.SecretKeySelector{
+ LocalObjectReference: corev1.LocalObjectReference{
+ Name: connectCredential.SecretName,
+ },
+ Key: keyName,
+ },
+ }})
+ }
+ }
+ appendEnvFromSecret(dptypes.DPDBUser, connectCredential.UsernameKey)
+ appendEnvFromSecret(dptypes.DPDBPassword, connectCredential.PasswordKey)
+ appendEnvFromSecret(dptypes.DPDBPort, connectCredential.PortKey)
+ if connectCredential.HostKey != "" {
+ appendEnvFromSecret(dptypes.DPDBHost, connectCredential.HostKey)
+ }
+ }
+ r.env = utils.MergeEnv(r.env, env)
+ return r
+}
+
+// builderRestoreJobName builds restore job name.
+func (r *restoreJobBuilder) builderRestoreJobName(jobIndex int) string {
+ jobName := fmt.Sprintf("restore-%s-%s-%s-%d", strings.ToLower(string(r.stage)), r.restore.UID[:8], r.backupSet.Backup.Name, jobIndex)
+ l := len(jobName)
+ if l > 63 {
+ return fmt.Sprintf("%s-%s", jobName[:57], jobName[l-5:l])
+ }
+ return jobName
+}
+
+// build the restore job by this builder.
+func (r *restoreJobBuilder) build(jobIndex int) *batchv1.Job {
+ job := &batchv1.Job{
+ ObjectMeta: metav1.ObjectMeta{
+ Name: r.builderRestoreJobName(jobIndex),
+ Namespace: r.restore.Namespace,
+ Labels: BuildRestoreLabels(r.restore.Name),
+ },
+ }
+ podSpec := job.Spec.Template.Spec
+ // 1. set pod spec
+ runUser := int64(0)
+ podSpec.SecurityContext = &corev1.PodSecurityContext{
+ RunAsUser: &runUser,
+ }
+ podSpec.RestartPolicy = corev1.RestartPolicyOnFailure
+ if r.stage == dpv1alpha1.PrepareData {
+ // set scheduling spec
+ schedulingSpec := r.restore.Spec.PrepareDataConfig.SchedulingSpec
+ podSpec.Tolerations = schedulingSpec.Tolerations
+ podSpec.Affinity = schedulingSpec.Affinity
+ podSpec.NodeSelector = schedulingSpec.NodeSelector
+ podSpec.NodeName = schedulingSpec.NodeName
+ podSpec.SchedulerName = schedulingSpec.SchedulerName
+ podSpec.TopologySpreadConstraints = schedulingSpec.TopologySpreadConstraints
+ } else {
+ podSpec.Tolerations = r.tolerations
+ podSpec.NodeSelector = r.nodeSelector
+ }
+ r.specificVolumes = append(r.specificVolumes, r.commonVolumes...)
+ podSpec.Volumes = r.specificVolumes
+ job.Spec.Template.Spec = podSpec
+ job.Spec.BackoffLimit = &defaultBackoffLimit
+
+ // 2. set restore container
+ r.specificVolumeMounts = append(r.specificVolumeMounts, r.commonVolumeMounts...)
+ container := corev1.Container{
+ Name: Restore,
+ Resources: r.restore.Spec.ContainerResources,
+ Env: r.env,
+ VolumeMounts: r.specificVolumeMounts,
+ Command: r.command,
+ Image: r.image,
+ ImagePullPolicy: corev1.PullIfNotPresent,
+ }
+ intctrlutil.InjectZeroResourcesLimitsIfEmpty(&container)
+ job.Spec.Template.Spec.Containers = []corev1.Container{container}
+ controllerutil.AddFinalizer(job, dptypes.DataProtectionFinalizerName)
+ return job
+}
diff --git a/internal/dataprotection/restore/manager.go b/internal/dataprotection/restore/manager.go
new file mode 100644
index 00000000000..cf0ae1dc768
--- /dev/null
+++ b/internal/dataprotection/restore/manager.go
@@ -0,0 +1,513 @@
+/*
+Copyright (C) 2022-2023 ApeCloud Co., Ltd
+
+This file is part of KubeBlocks project
+
+This program is free software: you can redistribute it and/or modify
+it under the terms of the GNU Affero General Public License as published by
+the Free Software Foundation, either version 3 of the License, or
+(at your option) any later version.
+
+This program is distributed in the hope that it will be useful
+but WITHOUT ANY WARRANTY; without even the implied warranty of
+MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+GNU Affero General Public License for more details.
+
+You should have received a copy of the GNU Affero General Public License
+along with this program. If not, see .
+*/
+
+package restore
+
+import (
+ "fmt"
+ "sort"
+
+ batchv1 "k8s.io/api/batch/v1"
+ corev1 "k8s.io/api/core/v1"
+ apierrors "k8s.io/apimachinery/pkg/api/errors"
+ metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
+ "k8s.io/apimachinery/pkg/runtime"
+ "k8s.io/apimachinery/pkg/types"
+ "k8s.io/client-go/tools/record"
+ "sigs.k8s.io/controller-runtime/pkg/client"
+ "sigs.k8s.io/controller-runtime/pkg/controller/controllerutil"
+
+ dpv1alpha1 "github.com/apecloud/kubeblocks/apis/dataprotection/v1alpha1"
+ "github.com/apecloud/kubeblocks/internal/constant"
+ intctrlutil "github.com/apecloud/kubeblocks/internal/controllerutil"
+ "github.com/apecloud/kubeblocks/internal/dataprotection/utils"
+ "github.com/apecloud/kubeblocks/internal/dataprotection/utils/boolptr"
+)
+
+type BackupActionSet struct {
+ Backup *dpv1alpha1.Backup
+ ActionSet *dpv1alpha1.ActionSet
+ UseVolumeSnapshot bool
+}
+
+type RestoreManager struct {
+ OriginalRestore *dpv1alpha1.Restore
+ Restore *dpv1alpha1.Restore
+ PrepareDataBackupSets []BackupActionSet
+ PostReadyBackupSets []BackupActionSet
+ Schema *runtime.Scheme
+ Recorder record.EventRecorder
+}
+
+func NewRestoreManager(restore *dpv1alpha1.Restore, recorder record.EventRecorder, schema *runtime.Scheme) *RestoreManager {
+ return &RestoreManager{
+ OriginalRestore: restore.DeepCopy(),
+ Restore: restore,
+ PrepareDataBackupSets: []BackupActionSet{},
+ PostReadyBackupSets: []BackupActionSet{},
+ Schema: schema,
+ Recorder: recorder,
+ }
+}
+
+// GetBackupActionSetByNamespaced gets the BackupActionSet by name and namespace of backup.
+func (r *RestoreManager) GetBackupActionSetByNamespaced(reqCtx intctrlutil.RequestCtx,
+ cli client.Client,
+ backupName,
+ namespace string) (*BackupActionSet, error) {
+ backup := &dpv1alpha1.Backup{}
+ if err := cli.Get(reqCtx.Ctx, types.NamespacedName{Namespace: namespace, Name: backupName}, backup); err != nil {
+ if apierrors.IsNotFound(err) {
+ err = intctrlutil.NewFatalError(err.Error())
+ }
+ return nil, err
+ }
+ backupMethod := backup.Status.BackupMethod
+ if backupMethod == nil {
+ return nil, intctrlutil.NewFatalError(fmt.Sprintf(`status.backupMethod of backup "%s" is empty`, backupName))
+ }
+ useVolumeSnapshot := backupMethod.SnapshotVolumes != nil && *backupMethod.SnapshotVolumes
+ actionSet, err := utils.GetActionSetByName(reqCtx, cli, backup.Status.BackupMethod.ActionSetName)
+ if err != nil {
+ return nil, err
+ }
+ return &BackupActionSet{Backup: backup, ActionSet: actionSet, UseVolumeSnapshot: useVolumeSnapshot}, nil
+}
+
+// BuildDifferentialBackupActionSets builds the backupActionSets for specified incremental backup.
+func (r *RestoreManager) BuildDifferentialBackupActionSets(reqCtx intctrlutil.RequestCtx, cli client.Client, sourceBackupSet BackupActionSet) error {
+ parentBackupSet, err := r.GetBackupActionSetByNamespaced(reqCtx, cli, sourceBackupSet.Backup.Spec.ParentBackupName, sourceBackupSet.Backup.Namespace)
+ if err != nil || parentBackupSet == nil {
+ return err
+ }
+ r.SetBackupSets(*parentBackupSet, sourceBackupSet)
+ return nil
+}
+
+// BuildIncrementalBackupActionSets builds the backupActionSets for specified incremental backup.
+func (r *RestoreManager) BuildIncrementalBackupActionSets(reqCtx intctrlutil.RequestCtx, cli client.Client, sourceBackupSet BackupActionSet) error {
+ r.SetBackupSets(sourceBackupSet)
+ if sourceBackupSet.ActionSet != nil && sourceBackupSet.ActionSet.Spec.BackupType == dpv1alpha1.BackupTypeIncremental {
+ // get the parent BackupActionSet for incremental.
+ backupSet, err := r.GetBackupActionSetByNamespaced(reqCtx, cli, sourceBackupSet.Backup.Spec.ParentBackupName, sourceBackupSet.Backup.Namespace)
+ if err != nil || backupSet == nil {
+ return err
+ }
+ return r.BuildIncrementalBackupActionSets(reqCtx, cli, *backupSet)
+ }
+ // if reaches full backup, sort the BackupActionSets and return
+ sortBackupSets := func(backupSets []BackupActionSet, reverse bool) []BackupActionSet {
+ sort.Slice(backupSets, func(i, j int) bool {
+ if reverse {
+ i, j = j, i
+ }
+ backupI := backupSets[i].Backup
+ backupJ := backupSets[j].Backup
+ if backupI == nil {
+ return false
+ }
+ if backupJ == nil {
+ return true
+ }
+ return compareWithBackupStopTime(*backupI, *backupJ)
+ })
+ return backupSets
+ }
+ r.PrepareDataBackupSets = sortBackupSets(r.PrepareDataBackupSets, false)
+ r.PostReadyBackupSets = sortBackupSets(r.PostReadyBackupSets, false)
+ return nil
+}
+
+func (r *RestoreManager) SetBackupSets(backupSets ...BackupActionSet) {
+ for i := range backupSets {
+ if backupSets[i].UseVolumeSnapshot {
+ r.PrepareDataBackupSets = append(r.PrepareDataBackupSets, backupSets[i])
+ continue
+ }
+ if backupSets[i].ActionSet == nil || backupSets[i].ActionSet.Spec.Restore == nil {
+ continue
+ }
+ if backupSets[i].ActionSet.Spec.Restore.PrepareData != nil {
+ r.PrepareDataBackupSets = append(r.PrepareDataBackupSets, backupSets[i])
+ }
+
+ if len(backupSets[i].ActionSet.Spec.Restore.PostReady) > 0 {
+ r.PostReadyBackupSets = append(r.PostReadyBackupSets, backupSets[i])
+ }
+ }
+}
+
+// AnalysisRestoreActionsWithBackup analysis the restore actions progress group by backup.
+// check if the restore jobs are completed or failed or processing.
+func (r *RestoreManager) AnalysisRestoreActionsWithBackup(stage dpv1alpha1.RestoreStage, backupName string, actionName string) (bool, bool) {
+ var (
+ restoreActionCount int
+ finishedActionCount int
+ exitFailedAction bool
+ )
+ restoreActions := r.Restore.Status.Actions.PostReady
+ if stage == dpv1alpha1.PrepareData {
+ restoreActions = r.Restore.Status.Actions.PrepareData
+ // if the stage is prepareData, actionCount keeps up with pvc count.
+ restoreActionCount = GetRestoreActionsCountForPrepareData(r.Restore.Spec.PrepareDataConfig)
+ }
+ for i := range restoreActions {
+ if restoreActions[i].BackupName != backupName && restoreActions[i].Name != actionName {
+ continue
+ }
+ // if the stage is PostReady, actionCount keeps up with actions
+ if stage == dpv1alpha1.PostReady {
+ restoreActionCount += 1
+ }
+ switch restoreActions[i].Status {
+ case dpv1alpha1.RestoreActionFailed:
+ finishedActionCount += 1
+ exitFailedAction = true
+ case dpv1alpha1.RestoreActionCompleted:
+ finishedActionCount += 1
+ }
+ }
+
+ allActionsFinished := restoreActionCount > 0 && finishedActionCount == restoreActionCount
+ return allActionsFinished, exitFailedAction
+}
+
+func (r *RestoreManager) RestorePVCFromSnapshot(reqCtx intctrlutil.RequestCtx, cli client.Client, backupSet BackupActionSet, actionName string) error {
+ prepareDataConfig := r.Restore.Spec.PrepareDataConfig
+ if prepareDataConfig == nil {
+ return nil
+ }
+ createPVCWithSnapshot := func(claim dpv1alpha1.RestoreVolumeClaim) error {
+ if claim.VolumeSource == "" {
+ return intctrlutil.NewFatalError(fmt.Sprintf(`claim "%s"" volumeSource can not be empty if the backup uses volume snapshot`, claim.Name))
+ }
+ // get volumeSnapshot by backup and volumeSource.
+ claim.VolumeClaimSpec.DataSource = &corev1.TypedLocalObjectReference{
+ Name: utils.GetBackupVolumeSnapshotName(backupSet.Backup.Name, claim.VolumeSource),
+ Kind: constant.VolumeSnapshotKind,
+ APIGroup: &volumeSnapshotGroup,
+ }
+ return r.createPVCIfNotExist(reqCtx, cli, claim.ObjectMeta, claim.VolumeClaimSpec)
+ }
+
+ for _, claim := range prepareDataConfig.RestoreVolumeClaims {
+ if err := createPVCWithSnapshot(claim); err != nil {
+ return err
+ }
+ }
+ claimTemplate := prepareDataConfig.RestoreVolumeClaimsTemplate
+
+ if claimTemplate != nil {
+ restoreJobReplicas := GetRestoreActionsCountForPrepareData(prepareDataConfig)
+ for i := 0; i < restoreJobReplicas; i++ {
+ // create pvc from claims template, build volumes and volumeMounts
+ for _, claim := range prepareDataConfig.RestoreVolumeClaimsTemplate.Templates {
+ claim.Name = fmt.Sprintf("%s-%d", claim.Name, i+int(claimTemplate.StartingIndex))
+ if err := createPVCWithSnapshot(claim); err != nil {
+ return err
+ }
+ }
+ }
+ }
+ // NOTE: do not to record status action for restoring from snapshot. it is not defined in ActionSet.
+ return nil
+}
+
+// BuildPrepareDataJobs builds the restore jobs for prepare pvc's data, and will create the target pvcs if not exist.
+func (r *RestoreManager) BuildPrepareDataJobs(reqCtx intctrlutil.RequestCtx, cli client.Client, backupSet BackupActionSet, actionName string) ([]*batchv1.Job, error) {
+ prepareDataConfig := r.Restore.Spec.PrepareDataConfig
+ if prepareDataConfig == nil {
+ return nil, nil
+ }
+ if !backupSet.ActionSet.HasPrepareDataStage() {
+ return nil, nil
+ }
+ jobBuilder := newRestoreJobBuilder(r.Restore, backupSet, dpv1alpha1.PrepareData).
+ setImage(backupSet.ActionSet.Spec.Restore.PrepareData.Image).
+ setCommand(backupSet.ActionSet.Spec.Restore.PrepareData.Command).
+ addBackupVolumeAndMount().
+ addCommonEnv()
+
+ createPVCIfNoteExistsAndBuildVolume := func(claim dpv1alpha1.RestoreVolumeClaim, identifier string) (*corev1.Volume, *corev1.VolumeMount, error) {
+ if err := r.createPVCIfNotExist(reqCtx, cli, claim.ObjectMeta, claim.VolumeClaimSpec); err != nil {
+ return nil, nil, err
+ }
+ return jobBuilder.buildPVCVolumeAndMount(claim, identifier)
+ }
+
+ // create pvc from volumeClaims, set volume and volumeMount to jobBuilder
+ for _, claim := range prepareDataConfig.RestoreVolumeClaims {
+ volume, volumeMount, err := createPVCIfNoteExistsAndBuildVolume(claim, "dp-claim")
+ if err != nil {
+ return nil, err
+ }
+ jobBuilder.addToCommonVolumesAndMounts(volume, volumeMount)
+ }
+
+ var (
+ restoreJobs []*batchv1.Job
+ restoreJobReplicas = GetRestoreActionsCountForPrepareData(prepareDataConfig)
+ claimsTemplate = prepareDataConfig.RestoreVolumeClaimsTemplate
+ )
+
+ if prepareDataConfig.IsSerialPolicy() {
+ // obtain the PVC serial number that needs to be restored
+ currentOrder := 1
+ prepareActions := r.Restore.Status.Actions.PrepareData
+ for i := range prepareActions {
+ if prepareActions[i].BackupName != backupSet.Backup.Name && prepareActions[i].Name != actionName {
+ continue
+ }
+ if prepareActions[i].Status == dpv1alpha1.RestoreActionCompleted && currentOrder < restoreJobReplicas {
+ currentOrder += 1
+ if prepareDataConfig.IsSerialPolicy() {
+ // if the restore policy is Serial, should delete the completed job to release the pvc.
+ if err := deleteRestoreJob(reqCtx, cli, prepareActions[i].ObjectKey, r.Restore.Namespace); err != nil {
+ return nil, err
+ }
+ }
+ }
+ }
+ restoreJobReplicas = currentOrder
+ }
+
+ // build restore job to prepare pvc's data
+ for i := 0; i < restoreJobReplicas; i++ {
+ // reset specific volumes and volumeMounts
+ jobBuilder.resetSpecificVolumesAndMounts()
+ if claimsTemplate != nil {
+ // create pvc from claims template, build volumes and volumeMounts
+ for _, claim := range claimsTemplate.Templates {
+ claim.Name = fmt.Sprintf("%s-%d", claim.Name, i+int(claimsTemplate.StartingIndex))
+ volume, volumeMount, err := createPVCIfNoteExistsAndBuildVolume(claim, "dp-claim-tpl")
+ if err != nil {
+ return nil, err
+ }
+ jobBuilder.addToSpecificVolumesAndMounts(volume, volumeMount)
+ }
+ }
+ // build job and append
+ job := jobBuilder.build(i)
+ if prepareDataConfig.IsSerialPolicy() &&
+ restoreJobHasCompleted(r.Restore.Status.Actions.PrepareData, job.Name) {
+ // if the job has completed and the restore policy is Serial, continue
+ continue
+ }
+ restoreJobs = append(restoreJobs, job)
+ }
+ return restoreJobs, nil
+}
+
+// BuildPostReadyActionJobs builds the post ready jobs.
+func (r *RestoreManager) BuildPostReadyActionJobs(reqCtx intctrlutil.RequestCtx, cli client.Client, backupSet BackupActionSet, actionSpec dpv1alpha1.ActionSpec) ([]*batchv1.Job, error) {
+ readyConfig := r.Restore.Spec.ReadyConfig
+ if readyConfig == nil {
+ return nil, nil
+ }
+ if !backupSet.ActionSet.HasPostReadyStage() {
+ return nil, nil
+ }
+ getTargetPodList := func(labelSelector metav1.LabelSelector, msgKey string) ([]corev1.Pod, error) {
+ targetPodList, err := utils.GetPodListByLabelSelector(reqCtx, cli, labelSelector)
+ if err != nil {
+ return nil, err
+ }
+ if len(targetPodList.Items) == 0 {
+ return nil, intctrlutil.NewFatalError(fmt.Sprintf("can not found any pod by spec.readyConfig.%s.target.podSelector", msgKey))
+ }
+ return targetPodList.Items, nil
+ }
+
+ jobBuilder := newRestoreJobBuilder(r.Restore, backupSet, dpv1alpha1.PostReady).
+ addBackupVolumeAndMount().
+ addCommonEnv()
+
+ buildJobsForJobAction := func() ([]*batchv1.Job, error) {
+ jobAction := r.Restore.Spec.ReadyConfig.JobAction
+ if jobAction == nil {
+ return nil, intctrlutil.NewFatalError("spec.readyConfig.jobAction can not be empty")
+ }
+ targetPodList, err := getTargetPodList(jobAction.Target.PodSelector, "jobAction")
+ if err != nil {
+ return nil, err
+ }
+ targetPod := targetPodList[0]
+ for _, volumeMount := range jobAction.Target.VolumeMounts {
+ for _, volume := range targetPod.Spec.Volumes {
+ if volume.Name != volumeMount.Name {
+ continue
+ }
+ jobBuilder.addToSpecificVolumesAndMounts(&volume, &volumeMount)
+ }
+ }
+ if boolptr.IsSetToTrue(actionSpec.Job.RunOnTargetPodNode) {
+ jobBuilder.setNodeNameToNodeSelector(targetPod.Spec.NodeName)
+ }
+ job := jobBuilder.setImage(actionSpec.Job.Image).
+ setCommand(actionSpec.Job.Command).
+ setToleration(targetPod.Spec.Tolerations).
+ addTargetPodAndCredentialEnv(&targetPod, r.Restore.Spec.ReadyConfig.ConnectCredential).build(0)
+ return []*batchv1.Job{job}, nil
+ }
+
+ buildJobsForExecAction := func() ([]*batchv1.Job, error) {
+ execAction := r.Restore.Spec.ReadyConfig.ExecAction
+ if execAction == nil {
+ return nil, intctrlutil.NewFatalError("spec.readyConfig.execAction can not be empty")
+ }
+ targetPodList, err := getTargetPodList(execAction.Target.PodSelector, "execAction")
+ if err != nil {
+ return nil, err
+ }
+ var restoreJobs []*batchv1.Job
+ for i := range targetPodList {
+ containerName := actionSpec.Exec.Container
+ if containerName == "" {
+ containerName = targetPodList[i].Spec.Containers[0].Name
+ }
+ command := fmt.Sprintf("kubectl -n %s exec -it pod/%s -c %s -- %s", targetPodList[i].Namespace, targetPodList[i].Name, containerName, actionSpec.Exec.Command)
+ jobBuilder.setImage(constant.KBToolsImage).setCommand([]string{"sh", "-c", command}).
+ setToleration(targetPodList[i].Spec.Tolerations).
+ addTargetPodAndCredentialEnv(&targetPodList[i], r.Restore.Spec.ReadyConfig.ConnectCredential)
+ restoreJobs = append(restoreJobs, jobBuilder.build(i))
+ }
+ return restoreJobs, nil
+ }
+
+ if actionSpec.Job != nil {
+ return buildJobsForJobAction()
+ }
+ return buildJobsForExecAction()
+}
+
+func (r *RestoreManager) createPVCIfNotExist(
+ reqCtx intctrlutil.RequestCtx,
+ cli client.Client,
+ claimMetadata metav1.ObjectMeta,
+ claimSpec corev1.PersistentVolumeClaimSpec) error {
+ claimMetadata.Namespace = reqCtx.Req.Namespace
+ pvc := &corev1.PersistentVolumeClaim{
+ ObjectMeta: claimMetadata,
+ Spec: claimSpec,
+ }
+ tmpPVC := &corev1.PersistentVolumeClaim{}
+ if err := cli.Get(reqCtx.Ctx, types.NamespacedName{Name: claimMetadata.Name, Namespace: claimMetadata.Namespace}, tmpPVC); err != nil {
+ if !apierrors.IsNotFound(err) {
+ return err
+ }
+ msg := fmt.Sprintf("created pvc %s/%s", pvc.Namespace, pvc.Name)
+ r.Recorder.Event(r.Restore, corev1.EventTypeNormal, reasonCreateRestorePVC, msg)
+ if err = cli.Create(reqCtx.Ctx, pvc); err != nil {
+ return client.IgnoreAlreadyExists(err)
+ }
+ }
+ return nil
+}
+
+// CreateJobsIfNotExist creates the jobs if not exist.
+func (r *RestoreManager) CreateJobsIfNotExist(reqCtx intctrlutil.RequestCtx,
+ cli client.Client,
+ objs []*batchv1.Job) ([]*batchv1.Job, error) {
+ // creates jobs if not exist
+ var fetchedJobs []*batchv1.Job
+ for i := range objs {
+ if objs[i] == nil {
+ continue
+ }
+ fetchedJob := &batchv1.Job{}
+ if err := cli.Get(reqCtx.Ctx, client.ObjectKeyFromObject(objs[i]), fetchedJob); err != nil {
+ if !apierrors.IsNotFound(err) {
+ return nil, err
+ }
+ if err = controllerutil.SetControllerReference(r.Restore, objs[i], r.Schema); err != nil {
+ return nil, err
+ }
+ if err = cli.Create(reqCtx.Ctx, objs[i]); err != nil && !apierrors.IsAlreadyExists(err) {
+ return nil, err
+ }
+ msg := fmt.Sprintf("created job %s/%s", objs[i].Namespace, objs[i].Name)
+ r.Recorder.Event(r.Restore, corev1.EventTypeNormal, reasonCreateRestoreJob, msg)
+ fetchedJobs = append(fetchedJobs, objs[i])
+ } else {
+ fetchedJobs = append(fetchedJobs, fetchedJob)
+ }
+ }
+ return fetchedJobs, nil
+}
+
+// CheckJobsDone checks if jobs are completed or failed.
+func (r *RestoreManager) CheckJobsDone(
+ stage dpv1alpha1.RestoreStage,
+ actionName string,
+ backupSet BackupActionSet,
+ fetchedJobs []*batchv1.Job) (bool, bool) {
+ var (
+ allJobFinished = true
+ existFailedJob bool
+ )
+ restoreActions := &r.Restore.Status.Actions.PrepareData
+ if stage == dpv1alpha1.PostReady {
+ restoreActions = &r.Restore.Status.Actions.PostReady
+ }
+ for i := range fetchedJobs {
+ statusAction := dpv1alpha1.RestoreStatusAction{
+ Name: actionName,
+ ObjectKey: buildJobKeyForActionStatus(fetchedJobs[i].Name),
+ BackupName: backupSet.Backup.Name,
+ }
+ if done, err := checkJobDone(fetchedJobs[i]); err != nil {
+ existFailedJob = true
+ statusAction.Status = dpv1alpha1.RestoreActionFailed
+ statusAction.Message = err.Error()
+ SetRestoreStatusAction(restoreActions, statusAction)
+ } else if done {
+ statusAction.Status = dpv1alpha1.RestoreActionCompleted
+ SetRestoreStatusAction(restoreActions, statusAction)
+ } else {
+ allJobFinished = false
+ statusAction.Status = dpv1alpha1.RestoreActionProcessing
+ SetRestoreStatusAction(restoreActions, statusAction)
+ }
+ }
+ return allJobFinished, existFailedJob
+}
+
+// Recalculation whether all actions have been completed.
+func (r *RestoreManager) Recalculation(backupName, actionName string, allActionsFinished, existFailedAction *bool) {
+ prepareDataConfig := r.Restore.Spec.PrepareDataConfig
+ if !prepareDataConfig.IsSerialPolicy() {
+ return
+ }
+
+ if *existFailedAction {
+ // under the Serial policy, restore will be failed if any action is failed.
+ *allActionsFinished = true
+ return
+ }
+ var actionCount int
+ for _, v := range r.Restore.Status.Actions.PrepareData {
+ if v.Name == actionName && v.BackupName == backupName {
+ actionCount += 1
+ }
+ }
+ if actionCount != GetRestoreActionsCountForPrepareData(prepareDataConfig) {
+ // if the number of actions is not equal to the number of target actions, the recovery has not yet ended
+ *allActionsFinished = false
+ }
+}
diff --git a/internal/dataprotection/restore/types.go b/internal/dataprotection/restore/types.go
new file mode 100644
index 00000000000..338fb3c1ab2
--- /dev/null
+++ b/internal/dataprotection/restore/types.go
@@ -0,0 +1,60 @@
+/*
+Copyright (C) 2022-2023 ApeCloud Co., Ltd
+
+This file is part of KubeBlocks project
+
+This program is free software: you can redistribute it and/or modify
+it under the terms of the GNU Affero General Public License as published by
+the Free Software Foundation, either version 3 of the License, or
+(at your option) any later version.
+
+This program is distributed in the hope that it will be useful
+but WITHOUT ANY WARRANTY; without even the implied warranty of
+MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+GNU Affero General Public License for more details.
+
+You should have received a copy of the GNU Affero General Public License
+along with this program. If not, see .
+*/
+
+package restore
+
+var volumeSnapshotGroup = "snapshot.storage.k8s.io"
+
+// Restore condition constants
+const (
+ // condition types
+ ConditionTypeRestoreValidationPassed = "ValidationPassed"
+ ConditionTypeRestorePreparedData = "PrepareData"
+ ConditionTypeReadinessProbe = "ReadinessProbe"
+ ConditionTypeRestorePostReady = "PostReady"
+
+ // condition reasons
+ ReasonRestoreStarting = "RestoreStarting"
+ ReasonRestoreCompleted = "RestoreCompleted"
+ ReasonRestoreFailed = "RestoreFailed"
+ ReasonValidateFailed = "ValidateFailed"
+ ReasonValidateSuccessfully = "ValidateSuccessfully"
+ ReasonProcessing = "Processing"
+ ReasonFailed = "Failed"
+ ReasonSucceed = "Succeed"
+ reasonCreateRestoreJob = "CreateRestoreJob"
+ reasonCreateRestorePVC = "CreateRestorePVC"
+)
+
+// labels key
+const (
+ DataProtectionLabelRestoreKey = "dataprotection.kubeblocks.io/restore"
+)
+
+// env name for restore
+
+const (
+ DPRestoreTime = "DP_RESTORE_TIME"
+ DPRestoreTimestamp = "DP_RESTORE_TIMESTAMP"
+)
+
+// Restore constant
+const Restore = "restore"
+
+var defaultBackoffLimit int32 = 3
diff --git a/internal/dataprotection/restore/utils.go b/internal/dataprotection/restore/utils.go
new file mode 100644
index 00000000000..d684a0a2419
--- /dev/null
+++ b/internal/dataprotection/restore/utils.go
@@ -0,0 +1,218 @@
+/*
+Copyright (C) 2022-2023 ApeCloud Co., Ltd
+
+This file is part of KubeBlocks project
+
+This program is free software: you can redistribute it and/or modify
+it under the terms of the GNU Affero General Public License as published by
+the Free Software Foundation, either version 3 of the License, or
+(at your option) any later version.
+
+This program is distributed in the hope that it will be useful
+but WITHOUT ANY WARRANTY; without even the implied warranty of
+MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+GNU Affero General Public License for more details.
+
+You should have received a copy of the GNU Affero General Public License
+along with this program. If not, see .
+*/
+
+package restore
+
+import (
+ "fmt"
+ "strings"
+ "time"
+
+ batchv1 "k8s.io/api/batch/v1"
+ corev1 "k8s.io/api/core/v1"
+ "k8s.io/apimachinery/pkg/api/meta"
+ metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
+ "k8s.io/apimachinery/pkg/types"
+ "sigs.k8s.io/controller-runtime/pkg/client"
+ "sigs.k8s.io/controller-runtime/pkg/controller/controllerutil"
+
+ dpv1alpha1 "github.com/apecloud/kubeblocks/apis/dataprotection/v1alpha1"
+ "github.com/apecloud/kubeblocks/internal/constant"
+ intctrlutil "github.com/apecloud/kubeblocks/internal/controllerutil"
+ dptypes "github.com/apecloud/kubeblocks/internal/dataprotection/types"
+)
+
+func SetRestoreCondition(restore *dpv1alpha1.Restore, status metav1.ConditionStatus, conditionType, reason, message string) {
+ condition := metav1.Condition{
+ Type: conditionType,
+ Reason: reason,
+ Message: message,
+ Status: status,
+ }
+ meta.SetStatusCondition(&restore.Status.Conditions, condition)
+}
+
+// SetRestoreValidationCondition sets restore condition which type is ConditionTypeRestoreValidationPassed.
+func SetRestoreValidationCondition(restore *dpv1alpha1.Restore, reason, message string) {
+ status := metav1.ConditionFalse
+ if reason == ReasonValidateSuccessfully {
+ status = metav1.ConditionTrue
+ }
+ SetRestoreCondition(restore, status, ConditionTypeRestoreValidationPassed, reason, message)
+}
+
+// SetRestoreStageCondition sets restore stage condition.
+func SetRestoreStageCondition(restore *dpv1alpha1.Restore, stage dpv1alpha1.RestoreStage, reason, message string) {
+ status := metav1.ConditionFalse
+ if reason == ReasonSucceed {
+ status = metav1.ConditionTrue
+ }
+ conditionType := ConditionTypeRestorePreparedData
+ if stage == dpv1alpha1.PostReady {
+ conditionType = ConditionTypeRestorePostReady
+ }
+ SetRestoreCondition(restore, status, conditionType, reason, message)
+}
+
+func FindRestoreStatusAction(actions []dpv1alpha1.RestoreStatusAction, key string) *dpv1alpha1.RestoreStatusAction {
+ for i := range actions {
+ if actions[i].ObjectKey == key {
+ return &actions[i]
+ }
+ }
+ return nil
+}
+
+func SetRestoreStatusAction(actions *[]dpv1alpha1.RestoreStatusAction,
+ statusAction dpv1alpha1.RestoreStatusAction) {
+ if actions == nil {
+ return
+ }
+ if statusAction.Message == "" {
+ switch statusAction.Status {
+ case dpv1alpha1.RestoreActionProcessing:
+ statusAction.Message = fmt.Sprintf(`"%s" is processing`, statusAction.ObjectKey)
+ case dpv1alpha1.RestoreActionCompleted:
+ statusAction.Message = fmt.Sprintf(`successfully processed the "%s"`, statusAction.ObjectKey)
+ case dpv1alpha1.RestoreActionFailed:
+ statusAction.Message = fmt.Sprintf(`"%s" is failed, you can describe it or logs the ownered pod to get more informations`, statusAction.ObjectKey)
+ }
+ }
+ if statusAction.Status != dpv1alpha1.RestoreActionProcessing {
+ statusAction.EndTime = metav1.Now()
+ }
+ existingAction := FindRestoreStatusAction(*actions, statusAction.ObjectKey)
+ if existingAction == nil {
+ statusAction.StartTime = metav1.Now()
+ *actions = append(*actions, statusAction)
+ return
+ }
+ if existingAction.Status != statusAction.Status {
+ existingAction.Status = statusAction.Status
+ existingAction.EndTime = statusAction.EndTime
+ existingAction.Message = statusAction.Message
+ }
+}
+
+func GetRestoreActionsCountForPrepareData(config *dpv1alpha1.PrepareDataConfig) int {
+ if config == nil {
+ return 0
+ }
+ count := 1
+ if config.RestoreVolumeClaimsTemplate != nil {
+ count = int(config.RestoreVolumeClaimsTemplate.Replicas)
+ }
+ return count
+}
+
+func BuildRestoreLabels(restoreName string) map[string]string {
+ return map[string]string{
+ constant.AppManagedByLabelKey: constant.AppName,
+ DataProtectionLabelRestoreKey: restoreName,
+ }
+}
+
+func GetRestoreDuration(status dpv1alpha1.RestoreStatus) *metav1.Duration {
+ if status.CompletionTimestamp == nil || status.StartTimestamp == nil {
+ return nil
+ }
+ return &metav1.Duration{Duration: status.CompletionTimestamp.Sub(status.StartTimestamp.Time).Round(time.Second)}
+}
+
+func getTimeFormat(envs []corev1.EnvVar) string {
+ for _, env := range envs {
+ if env.Name == dptypes.DPTimeFormat {
+ return env.Value
+ }
+ }
+ return time.RFC3339
+}
+
+// checkJobDone if the job is completed or failed, return true.
+// if the job is failed, return an error to describe the failed message.
+func checkJobDone(job *batchv1.Job) (bool, error) {
+ if job == nil {
+ return false, nil
+ }
+ for _, condition := range job.Status.Conditions {
+ if condition.Type == batchv1.JobComplete {
+ return true, nil
+ } else if condition.Type == batchv1.JobFailed {
+ return true, fmt.Errorf(condition.Reason + ": " + condition.Message)
+ }
+ }
+ return false, nil
+}
+
+func compareWithBackupStopTime(backupI, backupJ dpv1alpha1.Backup) bool {
+ endTimeI := backupI.GetEndTime()
+ endTimeJ := backupJ.GetEndTime()
+ if endTimeI.IsZero() {
+ return false
+ }
+ if endTimeJ.IsZero() {
+ return true
+ }
+ if endTimeI.Equal(endTimeJ) {
+ return backupI.Name < backupJ.Name
+ }
+ return endTimeI.Before(endTimeJ)
+}
+
+func buildJobKeyForActionStatus(jobName string) string {
+ return fmt.Sprintf("%s/%s", constant.JobKind, jobName)
+}
+
+func getMountPathWithSourceVolume(backup *dpv1alpha1.Backup, volumeSource string) string {
+ backupMethod := backup.Status.BackupMethod
+ if backupMethod != nil && backupMethod.TargetVolumes != nil {
+ for _, v := range backupMethod.TargetVolumes.VolumeMounts {
+ if v.Name == volumeSource {
+ return v.MountPath
+ }
+ }
+ }
+ return ""
+}
+
+func restoreJobHasCompleted(statusActions []dpv1alpha1.RestoreStatusAction, jobName string) bool {
+ jobKey := buildJobKeyForActionStatus(jobName)
+ for i := range statusActions {
+ if statusActions[i].ObjectKey == jobKey && statusActions[i].Status == dpv1alpha1.RestoreActionCompleted {
+ return true
+ }
+ }
+ return false
+}
+
+func deleteRestoreJob(reqCtx intctrlutil.RequestCtx, cli client.Client, jobKey string, namespace string) error {
+ jobName := strings.ReplaceAll(jobKey, fmt.Sprintf("%s/", constant.JobKind), "")
+ job := &batchv1.Job{}
+ if err := cli.Get(reqCtx.Ctx, types.NamespacedName{Name: jobName, Namespace: namespace}, job); err != nil {
+ return client.IgnoreNotFound(err)
+ }
+ if controllerutil.ContainsFinalizer(job, dptypes.DataProtectionFinalizerName) {
+ patch := client.MergeFrom(job.DeepCopy())
+ controllerutil.RemoveFinalizer(job, dptypes.DataProtectionFinalizerName)
+ if err := cli.Patch(reqCtx.Ctx, job, patch); err != nil {
+ return err
+ }
+ }
+ return intctrlutil.BackgroundDeleteObject(cli, reqCtx.Ctx, job)
+}
diff --git a/internal/dataprotection/types/constant.go b/internal/dataprotection/types/constant.go
new file mode 100644
index 00000000000..3254e3feb61
--- /dev/null
+++ b/internal/dataprotection/types/constant.go
@@ -0,0 +1,99 @@
+/*
+Copyright (C) 2022-2023 ApeCloud Co., Ltd
+
+This file is part of KubeBlocks project
+
+This program is free software: you can redistribute it and/or modify
+it under the terms of the GNU Affero General Public License as published by
+the Free Software Foundation, either version 3 of the License, or
+(at your option) any later version.
+
+This program is distributed in the hope that it will be useful
+but WITHOUT ANY WARRANTY; without even the implied warranty of
+MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+GNU Affero General Public License for more details.
+
+You should have received a copy of the GNU Affero General Public License
+along with this program. If not, see .
+*/
+
+package types
+
+const (
+ // DataProtectionFinalizerName is the name of our custom finalizer
+ DataProtectionFinalizerName = "dataprotection.kubeblocks.io/finalizer"
+)
+
+// annotation keys
+const (
+ // DefaultBackupPolicyAnnotationKey specifies the default backup policy.
+ DefaultBackupPolicyAnnotationKey = "dataprotection.kubeblocks.io/is-default-policy"
+ // DefaultBackupPolicyTemplateAnnotationKey specifies the default backup policy template.
+ DefaultBackupPolicyTemplateAnnotationKey = "dataprotection.kubeblocks.io/is-default-policy-template"
+ // DefaultBackupRepoAnnotationKey specifies the default backup repo.
+ DefaultBackupRepoAnnotationKey = "dataprotection.kubeblocks.io/is-default-repo"
+ // BackupDataPathPrefixAnnotationKey specifies the backup data path prefix.
+ BackupDataPathPrefixAnnotationKey = "dataprotection.kubeblocks.io/path-prefix"
+ // ReconfigureRefAnnotationKey specifies the reconfigure ref.
+ ReconfigureRefAnnotationKey = "dataprotection.kubeblocks.io/reconfigure-ref"
+)
+
+// label keys
+const (
+ // DataProtectionLabelClusterUIDKey specifies the cluster UID label key.
+ DataProtectionLabelClusterUIDKey = "dataprotection.kubeblocks.io/cluster-uid"
+ // BackupTypeLabelKeyKey specifies the backup type label key.
+ BackupTypeLabelKeyKey = "dataprotection.kubeblocks.io/backup-type"
+ // DataProtectionLabelBackupNameKey specifies the backup name label key.
+ DataProtectionLabelBackupNameKey = "dataprotection.kubeblocks.io/backup-name"
+ // DataProtectionLabelBackupScheduleKey specifies the backup schedule label key.
+ DataProtectionLabelBackupScheduleKey = "dataprotection.kubeblocks.io/backup-schedule"
+ // DataProtectionLabelBackupPolicyKey specifies the backup policy label key.
+ DataProtectionLabelBackupPolicyKey = "dataprotection.kubeblocks.io/backup-policy"
+ // DataProtectionLabelBackupMethodKey specifies the backup method label key.
+ DataProtectionLabelBackupMethodKey = "dataprotection.kubeblocks.io/backup-method"
+ // DataProtectionLabelBackupTypeKey specifies the backup type label key.
+ DataProtectionLabelBackupTypeKey = "dataprotection.kubeblocks.io/backup-type"
+ // DataProtectionLabelAutoBackupKey specifies the auto backup label key.
+ DataProtectionLabelAutoBackupKey = "dataprotection.kubeblocks.io/autobackup"
+)
+
+// env names
+const (
+ // DPDBHost database host for dataProtection
+ DPDBHost = "DP_DB_HOST"
+ // DPDBUser database user for dataProtection
+ DPDBUser = "DP_DB_USER"
+ // DPDBPassword database password for dataProtection
+ DPDBPassword = "DP_DB_PASSWORD"
+ // DPDBEndpoint database endpoint for dataProtection
+ DPDBEndpoint = "DP_DB_ENDPOINT"
+ // DPDBPort database port for dataProtection
+ DPDBPort = "DP_DB_PORT"
+ // DPTargetPodName the target pod name
+ DPTargetPodName = "DP_TARGET_POD_NAME"
+ // DPBackupDIR the dest directory for backup data
+ DPBackupDIR = "DP_BACKUP_DIR"
+ // DPBackupName backup CR name
+ DPBackupName = "DP_BACKUP_NAME"
+ // DPTTL backup time to live, reference the backupPolicy.spec.retention.ttl
+ DPTTL = "DP_TTL"
+ // DPCheckInterval check interval for continue backup
+ DPCheckInterval = "DP_CHECK_INTERVAL"
+ // DPBackupInfoFile the file name which retains the backup.status info
+ DPBackupInfoFile = "DP_BACKUP_INFO_FILE"
+ // DPTimeFormat golang time format string
+ DPTimeFormat = "TIME_FORMAT"
+ // DPVolumeDataDIR the volume data dir
+ DPVolumeDataDIR = "VOLUME_DATA_DIR" //
+ // DPKBRecoveryTime recovery time
+ DPKBRecoveryTime = "KB_RECOVERY_TIME" // recovery time
+ // DPKBRecoveryTimestamp recovery timestamp
+ DPKBRecoveryTimestamp = "KB_RECOVERY_TIMESTAMP" // recovery timestamp
+ // DPBaseBackupStartTime base backup start time for pitr
+ DPBaseBackupStartTime = "BASE_BACKUP_START_TIME" // base backup start time for pitr
+ // DPBaseBackupStartTimestamp base backup start timestamp for pitr
+ DPBaseBackupStartTimestamp = "BASE_BACKUP_START_TIMESTAMP" // base backup start timestamp for pitr
+ // DPBackupStopTime backup stop time
+ DPBackupStopTime = "BACKUP_STOP_TIME" // backup stop time
+)
diff --git a/internal/dataprotection/types/types.go b/internal/dataprotection/types/types.go
new file mode 100644
index 00000000000..a12b87b5eec
--- /dev/null
+++ b/internal/dataprotection/types/types.go
@@ -0,0 +1,25 @@
+/*
+Copyright (C) 2022-2023 ApeCloud Co., Ltd
+
+This file is part of KubeBlocks project
+
+This program is free software: you can redistribute it and/or modify
+it under the terms of the GNU Affero General Public License as published by
+the Free Software Foundation, either version 3 of the License, or
+(at your option) any later version.
+
+This program is distributed in the hope that it will be useful
+but WITHOUT ANY WARRANTY; without even the implied warranty of
+MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+GNU Affero General Public License for more details.
+
+You should have received a copy of the GNU Affero General Public License
+along with this program. If not, see .
+*/
+
+package types
+
+var (
+ // DefaultBackOffLimit is the default backoff limit for jobs.
+ DefaultBackOffLimit = int32(3)
+)
diff --git a/internal/dataprotection/utils/boolptr/boolptr.go b/internal/dataprotection/utils/boolptr/boolptr.go
new file mode 100644
index 00000000000..5ab816092ba
--- /dev/null
+++ b/internal/dataprotection/utils/boolptr/boolptr.go
@@ -0,0 +1,42 @@
+/*
+Copyright (C) 2022-2023 ApeCloud Co., Ltd
+
+This file is part of KubeBlocks project
+
+This program is free software: you can redistribute it and/or modify
+it under the terms of the GNU Affero General Public License as published by
+the Free Software Foundation, either version 3 of the License, or
+(at your option) any later version.
+
+This program is distributed in the hope that it will be useful
+but WITHOUT ANY WARRANTY; without even the implied warranty of
+MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+GNU Affero General Public License for more details.
+
+You should have received a copy of the GNU Affero General Public License
+along with this program. If not, see .
+*/
+
+package boolptr
+
+// IsSetToTrue returns true if and only if the bool pointer is non-nil and set to true.
+func IsSetToTrue(b *bool) bool {
+ return b != nil && *b
+}
+
+// IsSetToFalse returns true if and only if the bool pointer is non-nil and set to false.
+func IsSetToFalse(b *bool) bool {
+ return b != nil && !*b
+}
+
+// True returns a *bool whose underlying value is true.
+func True() *bool {
+ t := true
+ return &t
+}
+
+// False returns a *bool whose underlying value is false.
+func False() *bool {
+ t := false
+ return &t
+}
diff --git a/internal/dataprotection/utils/boolptr/boolptr_test.go b/internal/dataprotection/utils/boolptr/boolptr_test.go
new file mode 100644
index 00000000000..ff72c3c9733
--- /dev/null
+++ b/internal/dataprotection/utils/boolptr/boolptr_test.go
@@ -0,0 +1,39 @@
+/*
+Copyright (C) 2022-2023 ApeCloud Co., Ltd
+
+This file is part of KubeBlocks project
+
+This program is free software: you can redistribute it and/or modify
+it under the terms of the GNU Affero General Public License as published by
+the Free Software Foundation, either version 3 of the License, or
+(at your option) any later version.
+
+This program is distributed in the hope that it will be useful
+but WITHOUT ANY WARRANTY; without even the implied warranty of
+MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+GNU Affero General Public License for more details.
+
+You should have received a copy of the GNU Affero General Public License
+along with this program. If not, see .
+*/
+
+package boolptr
+
+import (
+ "testing"
+)
+
+func TestBoolPtr(t *testing.T) {
+ if !IsSetToTrue(True()) {
+ t.Errorf("True() should return a pointer to true")
+ }
+ if IsSetToTrue(False()) {
+ t.Errorf("False() should return a pointer to false")
+ }
+ if IsSetToFalse(True()) {
+ t.Errorf("True() should return a pointer to true")
+ }
+ if !IsSetToFalse(False()) {
+ t.Errorf("False() should return a pointer to false")
+ }
+}
diff --git a/internal/dataprotection/utils/envvar.go b/internal/dataprotection/utils/envvar.go
new file mode 100644
index 00000000000..8b1a529f2c4
--- /dev/null
+++ b/internal/dataprotection/utils/envvar.go
@@ -0,0 +1,63 @@
+/*
+Copyright (C) 2022-2023 ApeCloud Co., Ltd
+
+This file is part of KubeBlocks project
+
+This program is free software: you can redistribute it and/or modify
+it under the terms of the GNU Affero General Public License as published by
+the Free Software Foundation, either version 3 of the License, or
+(at your option) any later version.
+
+This program is distributed in the hope that it will be useful
+but WITHOUT ANY WARRANTY; without even the implied warranty of
+MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+GNU Affero General Public License for more details.
+
+You should have received a copy of the GNU Affero General Public License
+along with this program. If not, see .
+*/
+
+package utils
+
+import (
+ corev1 "k8s.io/api/core/v1"
+
+ dpv1alpha1 "github.com/apecloud/kubeblocks/apis/dataprotection/v1alpha1"
+ intctrlutil "github.com/apecloud/kubeblocks/internal/controllerutil"
+ dptypes "github.com/apecloud/kubeblocks/internal/dataprotection/types"
+)
+
+func BuildEnvByCredential(pod *corev1.Pod, credential *dpv1alpha1.ConnectionCredential) []corev1.EnvVar {
+ var envVars []corev1.EnvVar
+ if credential == nil {
+ return nil
+ }
+ var hostEnv corev1.EnvVar
+ if credential.HostKey == "" {
+ hostEnv = corev1.EnvVar{Name: dptypes.DPDBHost,
+ Value: intctrlutil.BuildPodHostDNS(pod)}
+ } else {
+ hostEnv = buildEnvBySecretKey(dptypes.DPDBHost, credential.SecretName, credential.HostKey)
+ }
+ envVars = append(envVars,
+ buildEnvBySecretKey(dptypes.DPDBUser, credential.SecretName, credential.UsernameKey),
+ buildEnvBySecretKey(dptypes.DPDBPassword, credential.SecretName, credential.PasswordKey),
+ buildEnvBySecretKey(dptypes.DPDBPort, credential.SecretName, credential.PortKey),
+ hostEnv,
+ )
+ return envVars
+}
+
+func buildEnvBySecretKey(name, secretName, key string) corev1.EnvVar {
+ return corev1.EnvVar{
+ Name: name,
+ ValueFrom: &corev1.EnvVarSource{
+ SecretKeyRef: &corev1.SecretKeySelector{
+ LocalObjectReference: corev1.LocalObjectReference{
+ Name: secretName,
+ },
+ Key: key,
+ },
+ },
+ }
+}
diff --git a/internal/dataprotection/utils/utils.go b/internal/dataprotection/utils/utils.go
new file mode 100644
index 00000000000..0da5fb69bbb
--- /dev/null
+++ b/internal/dataprotection/utils/utils.go
@@ -0,0 +1,135 @@
+/*
+Copyright (C) 2022-2023 ApeCloud Co., Ltd
+
+This file is part of KubeBlocks project
+
+This program is free software: you can redistribute it and/or modify
+it under the terms of the GNU Affero General Public License as published by
+the Free Software Foundation, either version 3 of the License, or
+(at your option) any later version.
+
+This program is distributed in the hope that it will be useful
+but WITHOUT ANY WARRANTY; without even the implied warranty of
+MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+GNU Affero General Public License for more details.
+
+You should have received a copy of the GNU Affero General Public License
+along with this program. If not, see .
+*/
+
+package utils
+
+import (
+ "context"
+ "fmt"
+
+ batchv1 "k8s.io/api/batch/v1"
+ corev1 "k8s.io/api/core/v1"
+ metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
+ "k8s.io/apimachinery/pkg/util/json"
+ "sigs.k8s.io/controller-runtime/pkg/client"
+ "sigs.k8s.io/controller-runtime/pkg/controller/controllerutil"
+
+ dpv1alpha1 "github.com/apecloud/kubeblocks/apis/dataprotection/v1alpha1"
+ "github.com/apecloud/kubeblocks/internal/constant"
+ intctrlutil "github.com/apecloud/kubeblocks/internal/controllerutil"
+ dptypes "github.com/apecloud/kubeblocks/internal/dataprotection/types"
+ viper "github.com/apecloud/kubeblocks/internal/viperx"
+)
+
+func AddTolerations(podSpec *corev1.PodSpec) (err error) {
+ if cmTolerations := viper.GetString(constant.CfgKeyCtrlrMgrTolerations); cmTolerations != "" {
+ if err = json.Unmarshal([]byte(cmTolerations), &podSpec.Tolerations); err != nil {
+ return err
+ }
+ }
+ if cmAffinity := viper.GetString(constant.CfgKeyCtrlrMgrAffinity); cmAffinity != "" {
+ if err = json.Unmarshal([]byte(cmAffinity), &podSpec.Affinity); err != nil {
+ return err
+ }
+ }
+ if cmNodeSelector := viper.GetString(constant.CfgKeyCtrlrMgrNodeSelector); cmNodeSelector != "" {
+ if err = json.Unmarshal([]byte(cmNodeSelector), &podSpec.NodeSelector); err != nil {
+ return err
+ }
+ }
+ return nil
+}
+
+func IsJobFinished(job *batchv1.Job) (bool, batchv1.JobConditionType, string) {
+ for _, c := range job.Status.Conditions {
+ if c.Type == batchv1.JobComplete && c.Status == corev1.ConditionTrue {
+ return true, c.Type, ""
+ }
+ if c.Type == batchv1.JobFailed && c.Status == corev1.ConditionTrue {
+ return true, c.Type, c.Reason + "/" + c.Message
+ }
+ }
+ return false, "", ""
+}
+
+func RemoveDataProtectionFinalizer(ctx context.Context, cli client.Client, obj client.Object) error {
+ if !controllerutil.ContainsFinalizer(obj, dptypes.DataProtectionFinalizerName) {
+ return nil
+ }
+ patch := client.MergeFrom(obj.DeepCopyObject().(client.Object))
+ controllerutil.RemoveFinalizer(obj, dptypes.DataProtectionFinalizerName)
+ return cli.Patch(ctx, obj, patch)
+}
+
+// GetActionSetByName gets the ActionSet by name.
+func GetActionSetByName(reqCtx intctrlutil.RequestCtx,
+ cli client.Client, name string) (*dpv1alpha1.ActionSet, error) {
+ if name == "" {
+ return nil, nil
+ }
+ as := &dpv1alpha1.ActionSet{}
+ if err := cli.Get(reqCtx.Ctx, client.ObjectKey{Name: name}, as); err != nil {
+ reqCtx.Log.Error(err, "failed to get ActionSet for backup.", "ActionSet", name)
+ return nil, err
+ }
+ return as, nil
+}
+
+func GetPodListByLabelSelector(reqCtx intctrlutil.RequestCtx,
+ cli client.Client,
+ labelSelector metav1.LabelSelector) (*corev1.PodList, error) {
+ selector, err := metav1.LabelSelectorAsSelector(&labelSelector)
+ if err != nil {
+ return nil, err
+ }
+ targetPodList := &corev1.PodList{}
+ if err = cli.List(reqCtx.Ctx, targetPodList,
+ client.InNamespace(reqCtx.Req.Namespace),
+ client.MatchingLabelsSelector{Selector: selector}); err != nil {
+ return nil, err
+ }
+ return targetPodList, nil
+}
+
+func GetBackupVolumeSnapshotName(backupName, volumeSource string) string {
+ return fmt.Sprintf("%s-%s", backupName, volumeSource)
+}
+
+// MergeEnv merges the targetEnv to original env. if original env exist the same name var, it will be replaced.
+func MergeEnv(originalEnv, targetEnv []corev1.EnvVar) []corev1.EnvVar {
+ if len(targetEnv) == 0 {
+ return originalEnv
+ }
+ originalEnvIndexMap := map[string]int{}
+ for i := range originalEnv {
+ originalEnvIndexMap[originalEnv[i].Name] = i
+ }
+ for i := range targetEnv {
+ if index, ok := originalEnvIndexMap[targetEnv[i].Name]; ok {
+ originalEnv[index] = targetEnv[i]
+ } else {
+ originalEnv = append(originalEnv, targetEnv[i])
+ }
+ }
+ return originalEnv
+}
+
+func VolumeSnapshotEnabled() bool {
+ return viper.GetBool("VOLUMESNAPSHOT")
+}
diff --git a/internal/generics/type.go b/internal/generics/type.go
index a5c7bd4627d..2adc22def64 100644
--- a/internal/generics/type.go
+++ b/internal/generics/type.go
@@ -32,7 +32,7 @@ import (
"sigs.k8s.io/controller-runtime/pkg/client"
appsv1alpha1 "github.com/apecloud/kubeblocks/apis/apps/v1alpha1"
- dataprotectionv1alpha1 "github.com/apecloud/kubeblocks/apis/dataprotection/v1alpha1"
+ dpv1alpha1 "github.com/apecloud/kubeblocks/apis/dataprotection/v1alpha1"
extensionsv1alpha1 "github.com/apecloud/kubeblocks/apis/extensions/v1alpha1"
storagev1alpha1 "github.com/apecloud/kubeblocks/apis/storage/v1alpha1"
workloads "github.com/apecloud/kubeblocks/apis/workloads/v1alpha1"
@@ -58,59 +58,77 @@ type PObjList[T Object, L ObjList[T]] interface {
}
// signature is used as an argument passed to generic functions for type deduction.
+// Goland IDE 2023.2.1 and 2023.2.2 code inspector has a bug to infer pointer type like PObject and PObjList from Object and ObjectList.
+// To workaround this bug, we also pass the pointer type to generic functions in signature.
-var SecretSignature = func(_ corev1.Secret, _ corev1.SecretList) {}
-var ServiceSignature = func(_ corev1.Service, _ corev1.ServiceList) {}
-var PersistentVolumeClaimSignature = func(_ corev1.PersistentVolumeClaim, _ corev1.PersistentVolumeClaimList) {}
-var PodSignature = func(_ corev1.Pod, _ corev1.PodList) {}
-var EventSignature = func(_ corev1.Event, _ corev1.EventList) {}
-var ConfigMapSignature = func(_ corev1.ConfigMap, _ corev1.ConfigMapList) {}
-var EndpointsSignature = func(_ corev1.Endpoints, _ corev1.EndpointsList) {}
+var SecretSignature = func(_ corev1.Secret, _ *corev1.Secret, _ corev1.SecretList, _ *corev1.SecretList) {}
+var ServiceSignature = func(_ corev1.Service, _ *corev1.Service, _ corev1.ServiceList, _ *corev1.ServiceList) {}
+var PersistentVolumeClaimSignature = func(_ corev1.PersistentVolumeClaim, _ *corev1.PersistentVolumeClaim, _ corev1.PersistentVolumeClaimList, _ *corev1.PersistentVolumeClaimList) {
+}
+var PodSignature = func(_ corev1.Pod, _ *corev1.Pod, _ corev1.PodList, _ *corev1.PodList) {}
+var EventSignature = func(_ corev1.Event, _ *corev1.Event, _ corev1.EventList, _ *corev1.EventList) {}
+var ConfigMapSignature = func(_ corev1.ConfigMap, _ *corev1.ConfigMap, _ corev1.ConfigMapList, _ *corev1.ConfigMapList) {}
+var EndpointsSignature = func(_ corev1.Endpoints, _ *corev1.Endpoints, _ corev1.EndpointsList, _ *corev1.EndpointsList) {}
-var RSMSignature = func(_ workloads.ReplicatedStateMachine, _ workloads.ReplicatedStateMachineList) {}
-var StatefulSetSignature = func(_ appsv1.StatefulSet, _ appsv1.StatefulSetList) {}
-var DeploymentSignature = func(_ appsv1.Deployment, _ appsv1.DeploymentList) {}
-var ReplicaSetSignature = func(_ appsv1.ReplicaSet, _ appsv1.ReplicaSetList) {}
+var RSMSignature = func(_ workloads.ReplicatedStateMachine, _ *workloads.ReplicatedStateMachine, _ workloads.ReplicatedStateMachineList, _ *workloads.ReplicatedStateMachineList) {
+}
+var StatefulSetSignature = func(_ appsv1.StatefulSet, _ *appsv1.StatefulSet, _ appsv1.StatefulSetList, _ *appsv1.StatefulSetList) {
+}
+var DeploymentSignature = func(_ appsv1.Deployment, _ *appsv1.Deployment, _ appsv1.DeploymentList, _ *appsv1.DeploymentList) {}
+var ReplicaSetSignature = func(_ appsv1.ReplicaSet, _ *appsv1.ReplicaSet, _ appsv1.ReplicaSetList, _ *appsv1.ReplicaSetList) {}
-var JobSignature = func(_ batchv1.Job, _ batchv1.JobList) {}
-var CronJobSignature = func(_ batchv1.CronJob, _ batchv1.CronJobList) {}
+var JobSignature = func(_ batchv1.Job, _ *batchv1.Job, _ batchv1.JobList, _ *batchv1.JobList) {}
+var CronJobSignature = func(_ batchv1.CronJob, _ *batchv1.CronJob, _ batchv1.CronJobList, _ *batchv1.CronJobList) {}
-var PodDisruptionBudgetSignature = func(_ policyv1.PodDisruptionBudget, _ policyv1.PodDisruptionBudgetList) {
+var PodDisruptionBudgetSignature = func(_ policyv1.PodDisruptionBudget, _ *policyv1.PodDisruptionBudget, _ policyv1.PodDisruptionBudgetList, _ *policyv1.PodDisruptionBudgetList) {
}
-var StorageClassSignature = func(_ storagev1.StorageClass, _ storagev1.StorageClassList) {}
-var CSIDriverSignature = func(_ storagev1.CSIDriver, _ storagev1.CSIDriverList) {}
+var StorageClassSignature = func(_ storagev1.StorageClass, _ *storagev1.StorageClass, _ storagev1.StorageClassList, _ *storagev1.StorageClassList) {
+}
+var CSIDriverSignature = func(_ storagev1.CSIDriver, _ *storagev1.CSIDriver, _ storagev1.CSIDriverList, _ *storagev1.CSIDriverList) {
+}
-var VolumeSnapshotSignature = func(_ snapshotv1.VolumeSnapshot, _ snapshotv1.VolumeSnapshotList) {}
+var VolumeSnapshotSignature = func(_ snapshotv1.VolumeSnapshot, _ *snapshotv1.VolumeSnapshot, _ snapshotv1.VolumeSnapshotList, _ *snapshotv1.VolumeSnapshotList) {
+}
-var ClusterSignature = func(_ appsv1alpha1.Cluster, _ appsv1alpha1.ClusterList) {}
-var ClusterVersionSignature = func(_ appsv1alpha1.ClusterVersion, _ appsv1alpha1.ClusterVersionList) {}
-var ClusterDefinitionSignature = func(_ appsv1alpha1.ClusterDefinition, _ appsv1alpha1.ClusterDefinitionList) {
+var ClusterSignature = func(_ appsv1alpha1.Cluster, _ *appsv1alpha1.Cluster, _ appsv1alpha1.ClusterList, _ *appsv1alpha1.ClusterList) {
+}
+var ClusterVersionSignature = func(_ appsv1alpha1.ClusterVersion, _ *appsv1alpha1.ClusterVersion, _ appsv1alpha1.ClusterVersionList, _ *appsv1alpha1.ClusterVersionList) {
+}
+var ClusterDefinitionSignature = func(_ appsv1alpha1.ClusterDefinition, _ *appsv1alpha1.ClusterDefinition, _ appsv1alpha1.ClusterDefinitionList, _ *appsv1alpha1.ClusterDefinitionList) {
}
-var OpsRequestSignature = func(_ appsv1alpha1.OpsRequest, _ appsv1alpha1.OpsRequestList) {}
-var ConfigConstraintSignature = func(_ appsv1alpha1.ConfigConstraint, _ appsv1alpha1.ConfigConstraintList) {
+var OpsRequestSignature = func(_ appsv1alpha1.OpsRequest, _ *appsv1alpha1.OpsRequest, _ appsv1alpha1.OpsRequestList, _ *appsv1alpha1.OpsRequestList) {
+}
+var ConfigConstraintSignature = func(_ appsv1alpha1.ConfigConstraint, _ *appsv1alpha1.ConfigConstraint, _ appsv1alpha1.ConfigConstraintList, _ *appsv1alpha1.ConfigConstraintList) {
}
-var BackupPolicyTemplateSignature = func(_ appsv1alpha1.BackupPolicyTemplate, _ appsv1alpha1.BackupPolicyTemplateList) {
+var BackupPolicyTemplateSignature = func(_ appsv1alpha1.BackupPolicyTemplate, _ *appsv1alpha1.BackupPolicyTemplate, _ appsv1alpha1.BackupPolicyTemplateList, _ *appsv1alpha1.BackupPolicyTemplateList) {
}
-var BackupPolicySignature = func(_ dataprotectionv1alpha1.BackupPolicy, _ dataprotectionv1alpha1.BackupPolicyList) {
+var BackupPolicySignature = func(_ dpv1alpha1.BackupPolicy, _ *dpv1alpha1.BackupPolicy, _ dpv1alpha1.BackupPolicyList, _ *dpv1alpha1.BackupPolicyList) {
}
-var BackupSignature = func(_ dataprotectionv1alpha1.Backup, _ dataprotectionv1alpha1.BackupList) {
+var BackupSignature = func(_ dpv1alpha1.Backup, _ *dpv1alpha1.Backup, _ dpv1alpha1.BackupList, _ *dpv1alpha1.BackupList) {
}
-var BackupToolSignature = func(_ dataprotectionv1alpha1.BackupTool, _ dataprotectionv1alpha1.BackupToolList) {
+var BackupScheduleSignature = func(_ dpv1alpha1.BackupSchedule, _ *dpv1alpha1.BackupSchedule, _ dpv1alpha1.BackupScheduleList, _ *dpv1alpha1.BackupScheduleList) {
}
-var RestoreJobSignature = func(_ dataprotectionv1alpha1.RestoreJob, _ dataprotectionv1alpha1.RestoreJobList) {
+var RestoreSignature = func(_ dpv1alpha1.Restore, _ *dpv1alpha1.Restore, _ dpv1alpha1.RestoreList, _ *dpv1alpha1.RestoreList) {
}
-var BackupRepoSignature = func(_ dataprotectionv1alpha1.BackupRepo, _ dataprotectionv1alpha1.BackupRepoList) {
+var ActionSetSignature = func(_ dpv1alpha1.ActionSet, _ *dpv1alpha1.ActionSet, _ dpv1alpha1.ActionSetList, _ *dpv1alpha1.ActionSetList) {
}
-var AddonSignature = func(_ extensionsv1alpha1.Addon, _ extensionsv1alpha1.AddonList) {
+var BackupRepoSignature = func(_ dpv1alpha1.BackupRepo, _ *dpv1alpha1.BackupRepo, _ dpv1alpha1.BackupRepoList, _ *dpv1alpha1.BackupRepoList) {
}
-var ComponentResourceConstraintSignature = func(_ appsv1alpha1.ComponentResourceConstraint, _ appsv1alpha1.ComponentResourceConstraintList) {}
-var ComponentClassDefinitionSignature = func(_ appsv1alpha1.ComponentClassDefinition, _ appsv1alpha1.ComponentClassDefinitionList) {}
-var StorageProviderSignature = func(_ storagev1alpha1.StorageProvider, _ storagev1alpha1.StorageProviderList) {}
+var AddonSignature = func(_ extensionsv1alpha1.Addon, _ *extensionsv1alpha1.Addon, _ extensionsv1alpha1.AddonList, _ *extensionsv1alpha1.AddonList) {
+}
+var ComponentResourceConstraintSignature = func(_ appsv1alpha1.ComponentResourceConstraint, _ *appsv1alpha1.ComponentResourceConstraint, _ appsv1alpha1.ComponentResourceConstraintList, _ *appsv1alpha1.ComponentResourceConstraintList) {
+}
+var ComponentClassDefinitionSignature = func(_ appsv1alpha1.ComponentClassDefinition, _ *appsv1alpha1.ComponentClassDefinition, _ appsv1alpha1.ComponentClassDefinitionList, _ *appsv1alpha1.ComponentClassDefinitionList) {
+}
-var ConfigurationSignature = func(_ appsv1alpha1.Configuration, _ appsv1alpha1.ConfigurationList) {}
+var StorageProviderSignature = func(_ storagev1alpha1.StorageProvider, _ *storagev1alpha1.StorageProvider, _ storagev1alpha1.StorageProviderList, _ *storagev1alpha1.StorageProviderList) {
+}
+
+var ConfigurationSignature = func(_ appsv1alpha1.Configuration, _ *appsv1alpha1.Configuration, _ appsv1alpha1.ConfigurationList, _ *appsv1alpha1.ConfigurationList) {
+}
func ToGVK(object client.Object) schema.GroupVersionKind {
t := reflect.TypeOf(object)
diff --git a/internal/gotemplate/functional_test.go b/internal/gotemplate/functional_test.go
new file mode 100644
index 00000000000..0f39ada1083
--- /dev/null
+++ b/internal/gotemplate/functional_test.go
@@ -0,0 +1,167 @@
+/*
+Copyright (C) 2022-2023 ApeCloud Co., Ltd
+
+This file is part of KubeBlocks project
+
+This program is free software: you can redistribute it and/or modify
+it under the terms of the GNU Affero General Public License as published by
+the Free Software Foundation, either version 3 of the License, or
+(at your option) any later version.
+
+This program is distributed in the hope that it will be useful
+but WITHOUT ANY WARRANTY; without even the implied warranty of
+MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+GNU Affero General Public License for more details.
+
+You should have received a copy of the GNU Affero General Public License
+along with this program. If not, see .
+*/
+
+package gotemplate
+
+import (
+ "reflect"
+ "testing"
+)
+
+func Test_regexStringSubmatch(t *testing.T) {
+ type args struct {
+ regex string
+ s string
+ }
+ tests := []struct {
+ name string
+ args args
+ want []string
+ wantErr bool
+ }{{
+ name: "test",
+ args: args{
+ regex: `^(\d+)K$`,
+ s: "123K",
+ },
+ want: []string{"123K", "123"},
+ wantErr: false,
+ }, {
+ name: "test",
+ args: args{
+ regex: `^(\d+)M$`,
+ s: "123",
+ },
+ want: nil,
+ wantErr: false,
+ }}
+ for _, tt := range tests {
+ t.Run(tt.name, func(t *testing.T) {
+ got, err := regexStringSubmatch(tt.args.regex, tt.args.s)
+ if (err != nil) != tt.wantErr {
+ t.Errorf("regexStringSubmatch() error = %v, wantErr %v", err, tt.wantErr)
+ return
+ }
+ if !reflect.DeepEqual(got, tt.want) {
+ t.Errorf("regexStringSubmatch() got = %v, want %v", got, tt.want)
+ }
+ })
+ }
+}
+
+func Test_fromYAML(t *testing.T) {
+ type args struct {
+ str string
+ }
+ tests := []struct {
+ name string
+ args args
+ want map[string]interface{}
+ wantErr bool
+ }{{
+ name: "test",
+ args: args{
+ str: ``,
+ },
+ want: map[string]interface{}{},
+ }, {
+ name: "test",
+ args: args{
+ str: `efg`,
+ },
+ want: map[string]interface{}{},
+ wantErr: true,
+ }, {
+ name: "test",
+ args: args{
+ str: `a:
+ b: "c"
+ c: "d"
+`,
+ },
+ want: map[string]interface{}{
+ "a": map[interface{}]interface{}{
+ "b": "c",
+ "c": "d",
+ },
+ },
+ }}
+ for _, tt := range tests {
+ t.Run(tt.name, func(t *testing.T) {
+ got, err := fromYAML(tt.args.str)
+ if (err != nil) != tt.wantErr {
+ t.Errorf("fromYAML() error = %v, wantErr %v", err, tt.wantErr)
+ return
+ }
+ if !reflect.DeepEqual(got, tt.want) {
+ t.Errorf("fromYAML() got = %v, want %v", got, tt.want)
+ }
+ })
+ }
+}
+
+func Test_fromYAMLArray(t *testing.T) {
+ type args struct {
+ str string
+ }
+ tests := []struct {
+ name string
+ args args
+ want []interface{}
+ wantErr bool
+ }{{
+ name: "test",
+ args: args{
+ str: ``,
+ },
+ want: nil,
+ }, {
+ name: "test",
+ args: args{
+ str: `abc: efg`,
+ },
+ wantErr: true,
+ }, {
+ name: "test",
+ args: args{
+ str: `
+- a
+- b
+- c
+`,
+ },
+ want: []interface{}{
+ "a",
+ "b",
+ "c",
+ },
+ }}
+ for _, tt := range tests {
+ t.Run(tt.name, func(t *testing.T) {
+ got, err := fromYAMLArray(tt.args.str)
+ if (err != nil) != tt.wantErr {
+ t.Errorf("fromYAMLArray() error = %v, wantErr %v", err, tt.wantErr)
+ return
+ }
+ if !reflect.DeepEqual(got, tt.want) {
+ t.Errorf("fromYAMLArray() got = %v, want %v", got, tt.want)
+ }
+ })
+ }
+}
diff --git a/internal/gotemplate/tpl_engine.go b/internal/gotemplate/tpl_engine.go
index 40cfe98dedb..6b73b713b00 100644
--- a/internal/gotemplate/tpl_engine.go
+++ b/internal/gotemplate/tpl_engine.go
@@ -74,6 +74,40 @@ type TplEngine struct {
ctx context.Context
}
+type TplEngineOptions func(*TplEngine)
+
+type DSLType string
+
+const (
+ DefaultDSL DSLType = "gotemplate"
+ KBDSL DSLType = "kbdsl"
+ KBDSL2 DSLType = "kbdsl2"
+
+ TemplateBeginDelim = "{{"
+ TemplateEndDelim = "}}"
+ KBDSLBeginDelim = "{%"
+ KBDSLEndDelim = "%}"
+ KBDSL2BeginDelim = "<%"
+ KBDSL2EndDelim = "%>"
+)
+
+func WithCustomizedSyntax(begin, end string) TplEngineOptions {
+ return func(t *TplEngine) {
+ t.tpl.Delims(begin, end)
+ }
+}
+
+func WithCustomizedWithType(dsl DSLType) TplEngineOptions {
+ switch dsl {
+ case KBDSL:
+ return WithCustomizedSyntax(KBDSLBeginDelim, KBDSLEndDelim)
+ case KBDSL2:
+ return WithCustomizedSyntax(KBDSL2BeginDelim, KBDSL2EndDelim)
+ default:
+ return WithCustomizedSyntax(TemplateBeginDelim, TemplateEndDelim)
+ }
+}
+
func (t *TplEngine) GetTplEngine() *template.Template {
return t.tpl
}
@@ -180,7 +214,7 @@ func (t *TplEngine) importSelfModuleFuncs(funcs map[string]functional, fn func(t
}
// NewTplEngine creates go template helper
-func NewTplEngine(values *TplValues, funcs *BuiltInObjectsFunc, tplName string, cli types2.ReadonlyClient, ctx context.Context) *TplEngine {
+func NewTplEngine(values *TplValues, funcs *BuiltInObjectsFunc, tplName string, cli types2.ReadonlyClient, ctx context.Context, options ...TplEngineOptions) *TplEngine {
coreBuiltinFuncs := sprig.TxtFuncMap()
// custom funcs
@@ -200,5 +234,8 @@ func NewTplEngine(values *TplValues, funcs *BuiltInObjectsFunc, tplName string,
}
engine.initSystemFunMap(coreBuiltinFuncs)
+ if len(options) > 0 {
+ options[0](&engine)
+ }
return &engine
}
diff --git a/internal/gotemplate/tpl_engine_test.go b/internal/gotemplate/tpl_engine_test.go
index a871bd0bc98..88596fdd325 100644
--- a/internal/gotemplate/tpl_engine_test.go
+++ b/internal/gotemplate/tpl_engine_test.go
@@ -240,4 +240,42 @@ mathAvg = [8-9][0-9]\.?\d*`
})
})
+ Context("customized syntax test", func() {
+ It("KB1 syntax", func() {
+ engine := NewTplEngine(&TplValues{}, nil, "for_test", nil, ctx, WithCustomizedWithType(KBDSL))
+ r, err := engine.Render(KBDSLBeginDelim + ` snakecase "getUserName" ` + KBDSLEndDelim)
+ Expect(err).Should(Succeed())
+ Expect("get_user_name").Should(BeEquivalentTo(r))
+ })
+
+ It("KB2 syntax", func() {
+ engine := NewTplEngine(&TplValues{}, nil, "for_test", nil, ctx, WithCustomizedWithType(KBDSL2))
+ r, err := engine.Render(KBDSL2BeginDelim + ` camelcase "get_user_name" ` + KBDSL2EndDelim)
+ Expect(err).Should(Succeed())
+ Expect("GetUserName").Should(BeEquivalentTo(r))
+ })
+
+ It("Default syntax", func() {
+ engine := NewTplEngine(&TplValues{}, nil, "for_test", nil, ctx, WithCustomizedWithType(DefaultDSL))
+ r, err := engine.Render(TemplateBeginDelim + ` camelcase "get_user_name" ` + TemplateEndDelim)
+ Expect(err).Should(Succeed())
+ Expect("GetUserName").Should(BeEquivalentTo(r))
+ })
+
+ It("customized syntax", func() {
+ engine := NewTplEngine(&TplValues{}, nil, "for_test", nil, ctx, WithCustomizedSyntax("-------", "======"))
+ r, err := engine.Render(`------- camelcase "get_user_name" ======`)
+ Expect(err).Should(Succeed())
+ Expect("GetUserName").Should(BeEquivalentTo(r))
+ })
+
+ It("default syntax", func() {
+ engine := NewTplEngine(&TplValues{}, nil, "for_test", nil, ctx)
+ r, _ := engine.Render(KBDSL2BeginDelim + ` snakecase "getUserName" ` + KBDSL2EndDelim)
+ Expect(`<% snakecase "getUserName" %>`).Should(BeEquivalentTo(r))
+ r, _ = engine.Render(KBDSLBeginDelim + ` camelcase "get_user_name" ` + KBDSLEndDelim)
+ Expect(`{% camelcase "get_user_name" %}`).Should(BeEquivalentTo(r))
+ })
+ })
+
})
diff --git a/internal/preflight/analyzer/kb_storage_class.go b/internal/preflight/analyzer/kb_storage_class.go
index 0f63d74e355..784cf4ee81d 100644
--- a/internal/preflight/analyzer/kb_storage_class.go
+++ b/internal/preflight/analyzer/kb_storage_class.go
@@ -33,7 +33,7 @@ import (
const (
StorageClassPath = "cluster-resources/storage-classes.json"
- StorageClassErrorPath = "cluster-resources/storage-classes-error.json"
+ StorageClassErrorPath = "cluster-resources/storage-classes-errors.json"
)
type AnalyzeStorageClassByKb struct {
@@ -68,8 +68,13 @@ func (a *AnalyzeStorageClassByKb) analyzeStorageClass(analyzer *preflightv1beta2
}
storageClassesErrorData, err := getFile(StorageClassErrorPath)
- if err != nil && storageClassesErrorData != nil && len(storageClassesErrorData) > 0 && len(storageClassesData) == 0 {
- return newWarnResultWithMessage(a.Title(), fmt.Sprintf("get nodes list from k8s failed, err:%v", err)), err
+ if err == nil && storageClassesErrorData != nil && len(storageClassesErrorData) > 0 && len(storageClassesData) == 0 {
+ var values []string
+ err = json.Unmarshal(storageClassesErrorData, &values)
+ if err != nil || len(values) == 0 {
+ return newWarnResultWithMessage(a.Title(), fmt.Sprintf("get storage class failed, err:%v", storageClassesErrorData)), err
+ }
+ return newWarnResultWithMessage(a.Title(), fmt.Sprintf("get storage class failed, err:%v", values[0])), nil
}
var storageClasses storagev1beta1.StorageClassList
diff --git a/internal/preflight/analyzer/kb_taint.go b/internal/preflight/analyzer/kb_taint.go
index 6989aa37f7f..6dddd47b1fd 100644
--- a/internal/preflight/analyzer/kb_taint.go
+++ b/internal/preflight/analyzer/kb_taint.go
@@ -38,7 +38,7 @@ import (
const (
NodesPath = "cluster-resources/nodes.json"
- NodesErrorPath = "cluster-resources/nodes-error.json"
+ NodesErrorPath = "cluster-resources/nodes-errors.json"
Tolerations = "tolerations"
KubeBlocks = "kubeblocks"
)
@@ -77,7 +77,12 @@ func (a *AnalyzeTaintClassByKb) analyzeTaint(getFile GetCollectedFileContents, f
nodesErrorData, err := getFile(NodesErrorPath)
if err != nil && nodesErrorData != nil && len(nodesErrorData) > 0 && len(nodesData) == 0 {
- return newFailedResultWithMessage(a.Title(), fmt.Sprintf("get nodes list from k8s failed, err:%v", err)), err
+ var values []string
+ err = json.Unmarshal(nodesErrorData, &values)
+ if err != nil || len(values) == 0 {
+ return newFailedResultWithMessage(a.Title(), fmt.Sprintf("get nodes from k8s failed, err:%v", nodesErrorData)), err
+ }
+ return newFailedResultWithMessage(a.Title(), fmt.Sprintf("get nodes from k8s failed, err:%v", values[0])), nil
}
var nodes v1.NodeList
diff --git a/internal/preflight/collect.go b/internal/preflight/collect.go
index 06dcaa11dd9..d41faa31330 100644
--- a/internal/preflight/collect.go
+++ b/internal/preflight/collect.go
@@ -24,7 +24,6 @@ import (
"encoding/json"
"fmt"
"reflect"
- "strings"
"time"
"github.com/pkg/errors"
@@ -328,7 +327,7 @@ func handleStorageClassError(ctx context.Context, _ preflight.CollectOpts, clien
return
}
var errorStrs []string
- if err := json.Unmarshal(storageClassError, &errorStrs); err != nil || len(errorStrs) == 0 || !isMetricsUnavailableError(errorStrs[0]) {
+ if err := json.Unmarshal(storageClassError, &errorStrs); err != nil || len(errorStrs) == 0 {
return
}
storageClasses, err := client.StorageV1().StorageClasses().List(ctx, metav1.ListOptions{})
@@ -354,10 +353,6 @@ func handleStorageClassError(ctx context.Context, _ preflight.CollectOpts, clien
data[StorageClassPath] = scBytes
}
-func isMetricsUnavailableError(str string) bool {
- return strings.Contains(str, "the server is currently unable to handle the request")
-}
-
func CollectRemoteData(ctx context.Context, preflightSpec *preflightv1beta2.HostPreflight, f cmdutil.Factory, progressCh chan interface{}) (*preflight.CollectResult, error) {
v := viper.GetViper()
diff --git a/internal/testutil/apps/backup_factory.go b/internal/testutil/apps/backup_factory.go
deleted file mode 100644
index 5695d1d00d9..00000000000
--- a/internal/testutil/apps/backup_factory.go
+++ /dev/null
@@ -1,70 +0,0 @@
-/*
-Copyright (C) 2022-2023 ApeCloud Co., Ltd
-
-This file is part of KubeBlocks project
-
-This program is free software: you can redistribute it and/or modify
-it under the terms of the GNU Affero General Public License as published by
-the Free Software Foundation, either version 3 of the License, or
-(at your option) any later version.
-
-This program is distributed in the hope that it will be useful
-but WITHOUT ANY WARRANTY; without even the implied warranty of
-MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
-GNU Affero General Public License for more details.
-
-You should have received a copy of the GNU Affero General Public License
-along with this program. If not, see .
-*/
-
-package apps
-
-import (
- "time"
-
- metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
-
- dataprotectionv1alpha1 "github.com/apecloud/kubeblocks/apis/dataprotection/v1alpha1"
-)
-
-type MockBackupFactory struct {
- BaseFactory[dataprotectionv1alpha1.Backup, *dataprotectionv1alpha1.Backup, MockBackupFactory]
-}
-
-func NewBackupFactory(namespace, name string) *MockBackupFactory {
- f := &MockBackupFactory{}
- f.init(namespace, name,
- &dataprotectionv1alpha1.Backup{
- Spec: dataprotectionv1alpha1.BackupSpec{},
- }, f)
- return f
-}
-
-func (factory *MockBackupFactory) SetBackupPolicyName(backupPolicyName string) *MockBackupFactory {
- factory.get().Spec.BackupPolicyName = backupPolicyName
- return factory
-}
-
-func (factory *MockBackupFactory) SetBackupType(backupType dataprotectionv1alpha1.BackupType) *MockBackupFactory {
- factory.get().Spec.BackupType = backupType
- return factory
-}
-
-func (factory *MockBackupFactory) SetLabels(labels map[string]string) *MockBackupFactory {
- factory.get().SetLabels(labels)
- return factory
-}
-
-func (factory *MockBackupFactory) SetBackLog(startTime, stopTime time.Time) *MockBackupFactory {
- manifests := factory.get().Status.Manifests
- if manifests == nil {
- manifests = &dataprotectionv1alpha1.ManifestsStatus{}
- }
- if manifests.BackupLog == nil {
- manifests.BackupLog = &dataprotectionv1alpha1.BackupLogStatus{}
- }
- manifests.BackupLog.StartTime = &metav1.Time{Time: startTime}
- manifests.BackupLog.StopTime = &metav1.Time{Time: stopTime}
- factory.get().Status.Manifests = manifests
- return factory
-}
diff --git a/internal/testutil/apps/backuppolicy_factory.go b/internal/testutil/apps/backuppolicy_factory.go
deleted file mode 100644
index 2b010383e6b..00000000000
--- a/internal/testutil/apps/backuppolicy_factory.go
+++ /dev/null
@@ -1,236 +0,0 @@
-/*
-Copyright (C) 2022-2023 ApeCloud Co., Ltd
-
-This file is part of KubeBlocks project
-
-This program is free software: you can redistribute it and/or modify
-it under the terms of the GNU Affero General Public License as published by
-the Free Software Foundation, either version 3 of the License, or
-(at your option) any later version.
-
-This program is distributed in the hope that it will be useful
-but WITHOUT ANY WARRANTY; without even the implied warranty of
-MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
-GNU Affero General Public License for more details.
-
-You should have received a copy of the GNU Affero General Public License
-along with this program. If not, see .
-*/
-
-package apps
-
-import (
- "k8s.io/apimachinery/pkg/api/resource"
- metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
- "k8s.io/utils/pointer"
-
- dataprotectionv1alpha1 "github.com/apecloud/kubeblocks/apis/dataprotection/v1alpha1"
- "github.com/apecloud/kubeblocks/internal/constant"
-)
-
-type MockBackupPolicyFactory struct {
- BaseFactory[dataprotectionv1alpha1.BackupPolicy, *dataprotectionv1alpha1.BackupPolicy, MockBackupPolicyFactory]
- backupType dataprotectionv1alpha1.BackupType
-}
-
-func NewBackupPolicyFactory(namespace, name string) *MockBackupPolicyFactory {
- f := &MockBackupPolicyFactory{}
- f.init(namespace, name,
- &dataprotectionv1alpha1.BackupPolicy{}, f)
- return f
-}
-
-func (factory *MockBackupPolicyFactory) setBasePolicyField(setField func(basePolicy *dataprotectionv1alpha1.BasePolicy)) {
- var basePolicy *dataprotectionv1alpha1.BasePolicy
- switch factory.backupType {
- case dataprotectionv1alpha1.BackupTypeDataFile:
- basePolicy = &factory.get().Spec.Datafile.BasePolicy
- case dataprotectionv1alpha1.BackupTypeLogFile:
- basePolicy = &factory.get().Spec.Logfile.BasePolicy
- case dataprotectionv1alpha1.BackupTypeSnapshot:
- basePolicy = &factory.get().Spec.Snapshot.BasePolicy
- }
- if basePolicy == nil {
- // ignore
- return
- }
- setField(basePolicy)
-}
-
-func (factory *MockBackupPolicyFactory) setCommonPolicyField(setField func(commonPolicy *dataprotectionv1alpha1.CommonBackupPolicy)) {
- var commonPolicy *dataprotectionv1alpha1.CommonBackupPolicy
- switch factory.backupType {
- case dataprotectionv1alpha1.BackupTypeDataFile:
- commonPolicy = factory.get().Spec.Datafile
- case dataprotectionv1alpha1.BackupTypeLogFile:
- commonPolicy = factory.get().Spec.Logfile
- }
- if commonPolicy == nil {
- // ignore
- return
- }
- setField(commonPolicy)
-}
-
-func (factory *MockBackupPolicyFactory) setScheduleField(setField func(schedulePolicy *dataprotectionv1alpha1.SchedulePolicy)) {
- var schedulePolicy *dataprotectionv1alpha1.SchedulePolicy
- switch factory.backupType {
- case dataprotectionv1alpha1.BackupTypeDataFile:
- factory.get().Spec.Schedule.Datafile = &dataprotectionv1alpha1.SchedulePolicy{}
- schedulePolicy = factory.get().Spec.Schedule.Datafile
- case dataprotectionv1alpha1.BackupTypeSnapshot:
- factory.get().Spec.Schedule.Snapshot = &dataprotectionv1alpha1.SchedulePolicy{}
- schedulePolicy = factory.get().Spec.Schedule.Snapshot
- case dataprotectionv1alpha1.BackupTypeLogFile:
- factory.get().Spec.Schedule.Logfile = &dataprotectionv1alpha1.SchedulePolicy{}
- schedulePolicy = factory.get().Spec.Schedule.Logfile
- }
- if schedulePolicy == nil {
- // ignore
- return
- }
- setField(schedulePolicy)
-}
-
-func (factory *MockBackupPolicyFactory) AddSnapshotPolicy() *MockBackupPolicyFactory {
- factory.get().Spec.Snapshot = &dataprotectionv1alpha1.SnapshotPolicy{
- Hooks: &dataprotectionv1alpha1.BackupPolicyHook{},
- }
- factory.backupType = dataprotectionv1alpha1.BackupTypeSnapshot
- return factory
-}
-
-func (factory *MockBackupPolicyFactory) AddDataFilePolicy() *MockBackupPolicyFactory {
- factory.get().Spec.Datafile = &dataprotectionv1alpha1.CommonBackupPolicy{
- PersistentVolumeClaim: dataprotectionv1alpha1.PersistentVolumeClaim{
- Name: pointer.String("backup-data"),
- CreatePolicy: dataprotectionv1alpha1.CreatePVCPolicyIfNotPresent,
- },
- }
- factory.backupType = dataprotectionv1alpha1.BackupTypeDataFile
- return factory
-}
-
-func (factory *MockBackupPolicyFactory) AddLogfilePolicy() *MockBackupPolicyFactory {
- factory.get().Spec.Logfile = &dataprotectionv1alpha1.CommonBackupPolicy{
- PersistentVolumeClaim: dataprotectionv1alpha1.PersistentVolumeClaim{
- Name: pointer.String("backup-data"),
- CreatePolicy: dataprotectionv1alpha1.CreatePVCPolicyIfNotPresent,
- },
- }
- factory.backupType = dataprotectionv1alpha1.BackupTypeLogFile
- return factory
-}
-
-func (factory *MockBackupPolicyFactory) SetBackupToolName(backupToolName string) *MockBackupPolicyFactory {
- factory.setCommonPolicyField(func(commonPolicy *dataprotectionv1alpha1.CommonBackupPolicy) {
- commonPolicy.BackupToolName = backupToolName
- })
- return factory
-}
-
-func (factory *MockBackupPolicyFactory) SetSchedule(schedule string, enable bool) *MockBackupPolicyFactory {
- factory.setScheduleField(func(schedulePolicy *dataprotectionv1alpha1.SchedulePolicy) {
- schedulePolicy.Enable = enable
- schedulePolicy.CronExpression = schedule
- })
- return factory
-}
-
-func (factory *MockBackupPolicyFactory) SetScheduleStartingDeadlineMinutes(startingDeadlineMinutes *int64) *MockBackupPolicyFactory {
- factory.get().Spec.Schedule.StartingDeadlineMinutes = startingDeadlineMinutes
- return factory
-}
-
-func (factory *MockBackupPolicyFactory) SetTTL(duration string) *MockBackupPolicyFactory {
- factory.get().Spec.Retention = &dataprotectionv1alpha1.RetentionSpec{
- TTL: &duration,
- }
- return factory
-}
-
-func (factory *MockBackupPolicyFactory) SetBackupsHistoryLimit(backupsHistoryLimit int32) *MockBackupPolicyFactory {
- factory.setBasePolicyField(func(basePolicy *dataprotectionv1alpha1.BasePolicy) {
- basePolicy.BackupsHistoryLimit = backupsHistoryLimit
- })
- return factory
-}
-
-func (factory *MockBackupPolicyFactory) AddMatchLabels(keyAndValues ...string) *MockBackupPolicyFactory {
- matchLabels := make(map[string]string)
- for k, v := range WithMap(keyAndValues...) {
- matchLabels[k] = v
- }
- factory.setBasePolicyField(func(basePolicy *dataprotectionv1alpha1.BasePolicy) {
- basePolicy.Target.LabelsSelector = &metav1.LabelSelector{
- MatchLabels: matchLabels,
- }
- })
- return factory
-}
-
-func (factory *MockBackupPolicyFactory) SetTargetSecretName(name string) *MockBackupPolicyFactory {
- factory.setBasePolicyField(func(basePolicy *dataprotectionv1alpha1.BasePolicy) {
- basePolicy.Target.Secret = &dataprotectionv1alpha1.BackupPolicySecret{Name: name}
- })
- return factory
-}
-
-func (factory *MockBackupPolicyFactory) SetHookContainerName(containerName string) *MockBackupPolicyFactory {
- snapshotPolicy := factory.get().Spec.Snapshot
- if snapshotPolicy == nil {
- return factory
- }
- snapshotPolicy.Hooks.ContainerName = containerName
- return factory
-}
-
-func (factory *MockBackupPolicyFactory) AddHookPreCommand(preCommand string) *MockBackupPolicyFactory {
- snapshotPolicy := factory.get().Spec.Snapshot
- if snapshotPolicy == nil {
- return factory
- }
- preCommands := &snapshotPolicy.Hooks.PreCommands
- *preCommands = append(*preCommands, preCommand)
- return factory
-}
-
-func (factory *MockBackupPolicyFactory) AddHookPostCommand(postCommand string) *MockBackupPolicyFactory {
- snapshotPolicy := factory.get().Spec.Snapshot
- if snapshotPolicy == nil {
- return factory
- }
- postCommands := &snapshotPolicy.Hooks.PostCommands
- *postCommands = append(*postCommands, postCommand)
- return factory
-}
-
-func (factory *MockBackupPolicyFactory) SetPVC(pvcName string) *MockBackupPolicyFactory {
- factory.setCommonPolicyField(func(commonPolicy *dataprotectionv1alpha1.CommonBackupPolicy) {
- if pvcName == "" {
- commonPolicy.PersistentVolumeClaim.Name = nil
- } else {
- commonPolicy.PersistentVolumeClaim.Name = &pvcName
- }
- commonPolicy.PersistentVolumeClaim.InitCapacity = resource.MustParse(constant.DefaultBackupPvcInitCapacity)
- })
- return factory
-}
-
-func (factory *MockBackupPolicyFactory) SetBackupRepo(repoName string) *MockBackupPolicyFactory {
- factory.setCommonPolicyField(func(commonPolicy *dataprotectionv1alpha1.CommonBackupPolicy) {
- if repoName == "" {
- commonPolicy.BackupRepoName = nil
- } else {
- commonPolicy.BackupRepoName = &repoName
- }
- })
- return factory
-}
-
-func (factory *MockBackupPolicyFactory) SetBackupStatusUpdates(backupStatusUpdates []dataprotectionv1alpha1.BackupStatusUpdate) *MockBackupPolicyFactory {
- factory.setBasePolicyField(func(basePolicy *dataprotectionv1alpha1.BasePolicy) {
- basePolicy.BackupStatusUpdates = backupStatusUpdates
- })
- return factory
-}
diff --git a/internal/testutil/apps/backuppolicytemplate_factory.go b/internal/testutil/apps/backuppolicytemplate_factory.go
index c0aef4858e0..b7a5026b5c5 100644
--- a/internal/testutil/apps/backuppolicytemplate_factory.go
+++ b/internal/testutil/apps/backuppolicytemplate_factory.go
@@ -20,195 +20,120 @@ along with this program. If not, see .
package apps
import (
+ corev1 "k8s.io/api/core/v1"
+
appsv1alpha1 "github.com/apecloud/kubeblocks/apis/apps/v1alpha1"
- dataprotectionv1alpha1 "github.com/apecloud/kubeblocks/apis/dataprotection/v1alpha1"
+ dpv1alpha1 "github.com/apecloud/kubeblocks/apis/dataprotection/v1alpha1"
)
type MockBackupPolicyTemplateFactory struct {
BaseFactory[appsv1alpha1.BackupPolicyTemplate, *appsv1alpha1.BackupPolicyTemplate, MockBackupPolicyTemplateFactory]
- backupType dataprotectionv1alpha1.BackupType
}
func NewBackupPolicyTemplateFactory(name string) *MockBackupPolicyTemplateFactory {
f := &MockBackupPolicyTemplateFactory{}
- f.init("", name,
+ f.Init("", name,
&appsv1alpha1.BackupPolicyTemplate{},
f)
return f
}
-func (factory *MockBackupPolicyTemplateFactory) SetClusterDefRef(clusterDefRef string) *MockBackupPolicyTemplateFactory {
- factory.get().Spec.ClusterDefRef = clusterDefRef
- return factory
+func (f *MockBackupPolicyTemplateFactory) SetClusterDefRef(clusterDefRef string) *MockBackupPolicyTemplateFactory {
+ f.Get().Spec.ClusterDefRef = clusterDefRef
+ return f
}
-func (factory *MockBackupPolicyTemplateFactory) getLastBackupPolicy() *appsv1alpha1.BackupPolicy {
- l := len(factory.get().Spec.BackupPolicies)
+func (f *MockBackupPolicyTemplateFactory) getLastBackupPolicy() *appsv1alpha1.BackupPolicy {
+ l := len(f.Get().Spec.BackupPolicies)
if l == 0 {
return nil
}
- backupPolicies := factory.get().Spec.BackupPolicies
+ backupPolicies := f.Get().Spec.BackupPolicies
return &backupPolicies[l-1]
}
-func (factory *MockBackupPolicyTemplateFactory) AddBackupPolicy(componentDef string) *MockBackupPolicyTemplateFactory {
- factory.get().Spec.BackupPolicies = append(factory.get().Spec.BackupPolicies, appsv1alpha1.BackupPolicy{
- ComponentDefRef: componentDef,
- })
- return factory
-}
-
-func (factory *MockBackupPolicyTemplateFactory) SetTTL(duration string) *MockBackupPolicyTemplateFactory {
- factory.getLastBackupPolicy().Retention = &appsv1alpha1.RetentionSpec{
- TTL: &duration,
+func (f *MockBackupPolicyTemplateFactory) getLastBackupMethod() *dpv1alpha1.BackupMethod {
+ backupPolicy := f.getLastBackupPolicy()
+ l := len(backupPolicy.BackupMethods)
+ if l == 0 {
+ return nil
}
- return factory
+ backupMethods := backupPolicy.BackupMethods
+ return &backupMethods[l-1]
}
-func (factory *MockBackupPolicyTemplateFactory) setBasePolicyField(setField func(basePolicy *appsv1alpha1.BasePolicy)) {
- backupPolicy := factory.getLastBackupPolicy()
- var basePolicy *appsv1alpha1.BasePolicy
- switch factory.backupType {
- case dataprotectionv1alpha1.BackupTypeDataFile:
- basePolicy = &backupPolicy.Datafile.BasePolicy
- case dataprotectionv1alpha1.BackupTypeLogFile:
- basePolicy = &backupPolicy.Logfile.BasePolicy
- case dataprotectionv1alpha1.BackupTypeSnapshot:
- basePolicy = &backupPolicy.Snapshot.BasePolicy
- }
- if basePolicy == nil {
- // ignore
- return
- }
- setField(basePolicy)
+func (f *MockBackupPolicyTemplateFactory) AddBackupPolicy(componentDef string) *MockBackupPolicyTemplateFactory {
+ f.Get().Spec.BackupPolicies = append(f.Get().Spec.BackupPolicies, appsv1alpha1.BackupPolicy{
+ ComponentDefRef: componentDef,
+ })
+ return f
}
-func (factory *MockBackupPolicyTemplateFactory) setCommonPolicyField(setField func(commonPolicy *appsv1alpha1.CommonBackupPolicy)) {
- backupPolicy := factory.getLastBackupPolicy()
- var commonPolicy *appsv1alpha1.CommonBackupPolicy
- switch factory.backupType {
- case dataprotectionv1alpha1.BackupTypeDataFile:
- commonPolicy = backupPolicy.Datafile
- case dataprotectionv1alpha1.BackupTypeLogFile:
- commonPolicy = backupPolicy.Logfile
- }
- if commonPolicy == nil {
- // ignore
- return
- }
- setField(commonPolicy)
+func (f *MockBackupPolicyTemplateFactory) SetRetentionPeriod(duration string) *MockBackupPolicyTemplateFactory {
+ f.getLastBackupPolicy().RetentionPeriod = dpv1alpha1.RetentionPeriod(duration)
+ return f
}
-func (factory *MockBackupPolicyTemplateFactory) setScheduleField(setField func(schedulePolicy *appsv1alpha1.SchedulePolicy)) {
- backupPolicy := factory.getLastBackupPolicy()
- var schedulePolicy *appsv1alpha1.SchedulePolicy
- switch factory.backupType {
- case dataprotectionv1alpha1.BackupTypeSnapshot:
- backupPolicy.Schedule.Snapshot = &appsv1alpha1.SchedulePolicy{}
- schedulePolicy = backupPolicy.Schedule.Snapshot
- case dataprotectionv1alpha1.BackupTypeDataFile:
- backupPolicy.Schedule.Datafile = &appsv1alpha1.SchedulePolicy{}
- schedulePolicy = backupPolicy.Schedule.Datafile
- case dataprotectionv1alpha1.BackupTypeLogFile:
- backupPolicy.Schedule.Logfile = &appsv1alpha1.SchedulePolicy{}
- schedulePolicy = backupPolicy.Schedule.Logfile
- }
- if schedulePolicy == nil {
+func (f *MockBackupPolicyTemplateFactory) setBackupPolicyField(setField func(backupPolicy *appsv1alpha1.BackupPolicy)) {
+ backupPolicy := f.getLastBackupPolicy()
+ if backupPolicy == nil {
// ignore
return
}
- setField(schedulePolicy)
+ setField(backupPolicy)
}
-func (factory *MockBackupPolicyTemplateFactory) AddSnapshotPolicy() *MockBackupPolicyTemplateFactory {
- backupPolicy := factory.getLastBackupPolicy()
- backupPolicy.Snapshot = &appsv1alpha1.SnapshotPolicy{
- Hooks: &appsv1alpha1.BackupPolicyHook{},
+func (f *MockBackupPolicyTemplateFactory) AddSchedule(method, schedule string, enable bool) *MockBackupPolicyTemplateFactory {
+ schedulePolicy := appsv1alpha1.SchedulePolicy{
+ Enabled: &enable,
+ CronExpression: schedule,
+ BackupMethod: method,
}
- factory.backupType = dataprotectionv1alpha1.BackupTypeSnapshot
- return factory
-}
-
-func (factory *MockBackupPolicyTemplateFactory) AddDatafilePolicy() *MockBackupPolicyTemplateFactory {
- backupPolicy := factory.getLastBackupPolicy()
- backupPolicy.Datafile = &appsv1alpha1.CommonBackupPolicy{}
- factory.backupType = dataprotectionv1alpha1.BackupTypeDataFile
- return factory
-}
-
-func (factory *MockBackupPolicyTemplateFactory) AddIncrementalPolicy() *MockBackupPolicyTemplateFactory {
- backupPolicy := factory.getLastBackupPolicy()
- backupPolicy.Logfile = &appsv1alpha1.CommonBackupPolicy{}
- factory.backupType = dataprotectionv1alpha1.BackupTypeLogFile
- return factory
+ backupPolicy := f.getLastBackupPolicy()
+ backupPolicy.Schedules = append(backupPolicy.Schedules, schedulePolicy)
+ return f
}
-func (factory *MockBackupPolicyTemplateFactory) SetHookContainerName(containerName string) *MockBackupPolicyTemplateFactory {
- backupPolicy := factory.getLastBackupPolicy()
- if backupPolicy.Snapshot == nil {
- return factory
- }
- backupPolicy.Snapshot.Hooks.ContainerName = containerName
- return factory
+func (f *MockBackupPolicyTemplateFactory) AddBackupMethod(name string,
+ snapshotVolumes bool, actionSetName string) *MockBackupPolicyTemplateFactory {
+ backupPolicy := f.getLastBackupPolicy()
+ backupPolicy.BackupMethods = append(backupPolicy.BackupMethods,
+ dpv1alpha1.BackupMethod{
+ Name: name,
+ SnapshotVolumes: &snapshotVolumes,
+ ActionSetName: actionSetName,
+ TargetVolumes: &dpv1alpha1.TargetVolumeInfo{},
+ })
+ return f
}
-func (factory *MockBackupPolicyTemplateFactory) AddHookPreCommand(preCommand string) *MockBackupPolicyTemplateFactory {
- backupPolicy := factory.getLastBackupPolicy()
- if backupPolicy.Snapshot == nil {
- return factory
- }
- preCommands := &backupPolicy.Snapshot.Hooks.PreCommands
- *preCommands = append(*preCommands, preCommand)
- return factory
+func (f *MockBackupPolicyTemplateFactory) SetBackupMethodVolumes(names []string) *MockBackupPolicyTemplateFactory {
+ backupMethod := f.getLastBackupMethod()
+ backupMethod.TargetVolumes.Volumes = names
+ return f
}
-func (factory *MockBackupPolicyTemplateFactory) AddHookPostCommand(postCommand string) *MockBackupPolicyTemplateFactory {
- backupPolicy := factory.getLastBackupPolicy()
- if backupPolicy.Snapshot == nil {
- return factory
+func (f *MockBackupPolicyTemplateFactory) SetBackupMethodVolumeMounts(keyAndValues ...string) *MockBackupPolicyTemplateFactory {
+ var volumeMounts []corev1.VolumeMount
+ for k, v := range WithMap(keyAndValues...) {
+ volumeMounts = append(volumeMounts, corev1.VolumeMount{
+ Name: k,
+ MountPath: v,
+ })
}
- postCommands := &backupPolicy.Snapshot.Hooks.PostCommands
- *postCommands = append(*postCommands, postCommand)
- return factory
-}
-
-func (factory *MockBackupPolicyTemplateFactory) SetSchedule(schedule string, enable bool) *MockBackupPolicyTemplateFactory {
- factory.setScheduleField(func(schedulePolicy *appsv1alpha1.SchedulePolicy) {
- schedulePolicy.Enable = enable
- schedulePolicy.CronExpression = schedule
- })
- return factory
-}
-
-func (factory *MockBackupPolicyTemplateFactory) SetBackupsHistoryLimit(backupsHistoryLimit int32) *MockBackupPolicyTemplateFactory {
- factory.setBasePolicyField(func(basePolicy *appsv1alpha1.BasePolicy) {
- basePolicy.BackupsHistoryLimit = backupsHistoryLimit
- })
- return factory
-}
-
-func (factory *MockBackupPolicyTemplateFactory) SetBackupToolName(backupToolName string) *MockBackupPolicyTemplateFactory {
- factory.setCommonPolicyField(func(commonPolicy *appsv1alpha1.CommonBackupPolicy) {
- commonPolicy.BackupToolName = backupToolName
- })
- return factory
-}
-
-func (factory *MockBackupPolicyTemplateFactory) SetTargetRole(role string) *MockBackupPolicyTemplateFactory {
- factory.setBasePolicyField(func(basePolicy *appsv1alpha1.BasePolicy) {
- basePolicy.Target.Role = role
- })
- return factory
+ backupMethod := f.getLastBackupMethod()
+ backupMethod.TargetVolumes.VolumeMounts = volumeMounts
+ return f
}
-func (factory *MockBackupPolicyTemplateFactory) SetTargetAccount(account string) *MockBackupPolicyTemplateFactory {
- factory.setBasePolicyField(func(basePolicy *appsv1alpha1.BasePolicy) {
- basePolicy.Target.Account = account
+func (f *MockBackupPolicyTemplateFactory) SetTargetRole(role string) *MockBackupPolicyTemplateFactory {
+ f.setBackupPolicyField(func(backupPolicy *appsv1alpha1.BackupPolicy) {
+ backupPolicy.Target.Role = role
})
- return factory
+ return f
}
-func (factory *MockBackupPolicyTemplateFactory) SetLabels(labels map[string]string) *MockBackupPolicyTemplateFactory {
- factory.get().SetLabels(labels)
- return factory
+func (f *MockBackupPolicyTemplateFactory) SetLabels(labels map[string]string) *MockBackupPolicyTemplateFactory {
+ f.Get().SetLabels(labels)
+ return f
}
diff --git a/internal/testutil/apps/base_factory.go b/internal/testutil/apps/base_factory.go
index c1bff9d2751..8cc3aeb3d19 100644
--- a/internal/testutil/apps/base_factory.go
+++ b/internal/testutil/apps/base_factory.go
@@ -40,7 +40,7 @@ type BaseFactory[T intctrlutil.Object, PT intctrlutil.PObject[T], F any] struct
concreteFactory *F
}
-func (factory *BaseFactory[T, PT, F]) init(namespace, name string, obj PT, f *F) {
+func (factory *BaseFactory[T, PT, F]) Init(namespace, name string, obj PT, f *F) {
obj.SetNamespace(namespace)
obj.SetName(name)
if obj.GetLabels() == nil {
@@ -53,7 +53,7 @@ func (factory *BaseFactory[T, PT, F]) init(namespace, name string, obj PT, f *F)
factory.concreteFactory = f
}
-func (factory *BaseFactory[T, PT, F]) get() PT {
+func (factory *BaseFactory[T, PT, F]) Get() PT {
return factory.object
}
@@ -89,7 +89,7 @@ func (factory *BaseFactory[T, PT, F]) AddAppComponentLabel(value string) *F {
return factory.AddLabels(constant.KBAppComponentLabelKey, value)
}
-func (factory *BaseFactory[T, PT, F]) AddAppManangedByLabel() *F {
+func (factory *BaseFactory[T, PT, F]) AddAppManagedByLabel() *F {
return factory.AddLabels(constant.AppManagedByLabelKey, constant.AppName)
}
@@ -137,22 +137,24 @@ func (factory *BaseFactory[T, PT, F]) AddFinalizers(finalizers []string) *F {
}
func (factory *BaseFactory[T, PT, F]) Apply(changeFn func(PT)) *F {
- changeFn(factory.object)
+ if changeFn != nil {
+ changeFn(factory.object)
+ }
return factory.concreteFactory
}
func (factory *BaseFactory[T, PT, F]) Create(testCtx *testutil.TestContext) *F {
- gomega.Expect(testCtx.CreateObj(testCtx.Ctx, factory.get())).Should(gomega.Succeed())
+ gomega.Expect(testCtx.CreateObj(testCtx.Ctx, factory.Get())).Should(gomega.Succeed())
return factory.concreteFactory
}
func (factory *BaseFactory[T, PT, F]) CheckedCreate(testCtx *testutil.TestContext) *F {
- gomega.Expect(testCtx.CheckedCreateObj(testCtx.Ctx, factory.get())).Should(gomega.Succeed())
+ gomega.Expect(testCtx.CheckedCreateObj(testCtx.Ctx, factory.Get())).Should(gomega.Succeed())
return factory.concreteFactory
}
func (factory *BaseFactory[T, PT, F]) CreateCli(ctx context.Context, cli client.Client) *F {
- gomega.Expect(cli.Create(ctx, factory.get())).Should(gomega.Succeed())
+ gomega.Expect(cli.Create(ctx, factory.Get())).Should(gomega.Succeed())
return factory.concreteFactory
}
diff --git a/internal/testutil/apps/cluster_consensus_test_util.go b/internal/testutil/apps/cluster_consensus_test_util.go
index 05e83a9fd6c..8adfa10e59d 100644
--- a/internal/testutil/apps/cluster_consensus_test_util.go
+++ b/internal/testutil/apps/cluster_consensus_test_util.go
@@ -120,11 +120,11 @@ func MockConsensusComponentStsPod(
if sts != nil {
stsUpdateRevision = sts.Status.UpdateRevision
}
- pod := NewPodFactory(testCtx.DefaultNamespace, podName).
+ podFactory := NewPodFactory(testCtx.DefaultNamespace, podName).
SetOwnerReferences("apps/v1", constant.StatefulSetKind, sts).
AddAppInstanceLabel(clusterName).
AddAppComponentLabel(consensusCompName).
- AddAppManangedByLabel().
+ AddAppManagedByLabel().
AddRoleLabel(podRole).
AddConsensusSetAccessModeLabel(accessMode).
AddControllerRevisionHashLabel(stsUpdateRevision).
@@ -149,8 +149,11 @@ func MockConsensusComponentStsPod(
},
},
},
- }).
- CheckedCreate(testCtx).GetObject()
+ })
+ if sts != nil && sts.Labels[constant.AppNameLabelKey] != "" {
+ podFactory.AddAppNameLabel(sts.Labels[constant.AppNameLabelKey])
+ }
+ pod := podFactory.CheckedCreate(testCtx).GetObject()
patch := client.MergeFrom(pod.DeepCopy())
pod.Status.Conditions = []corev1.PodCondition{
{
diff --git a/internal/testutil/apps/cluster_factory.go b/internal/testutil/apps/cluster_factory.go
index b7c578ad341..edea542bc4e 100644
--- a/internal/testutil/apps/cluster_factory.go
+++ b/internal/testutil/apps/cluster_factory.go
@@ -20,14 +20,9 @@ along with this program. If not, see .
package apps
import (
- "fmt"
- "time"
-
corev1 "k8s.io/api/core/v1"
- metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
appsv1alpha1 "github.com/apecloud/kubeblocks/apis/apps/v1alpha1"
- "github.com/apecloud/kubeblocks/internal/constant"
)
type MockClusterFactory struct {
@@ -36,7 +31,7 @@ type MockClusterFactory struct {
func NewClusterFactory(namespace, name, cdRef, cvRef string) *MockClusterFactory {
f := &MockClusterFactory{}
- f.init(namespace, name,
+ f.Init(namespace, name,
&appsv1alpha1.Cluster{
Spec: appsv1alpha1.ClusterSpec{
ClusterDefRef: cdRef,
@@ -49,17 +44,17 @@ func NewClusterFactory(namespace, name, cdRef, cvRef string) *MockClusterFactory
}
func (factory *MockClusterFactory) SetClusterAffinity(affinity *appsv1alpha1.Affinity) *MockClusterFactory {
- factory.get().Spec.Affinity = affinity
+ factory.Get().Spec.Affinity = affinity
return factory
}
func (factory *MockClusterFactory) AddClusterToleration(toleration corev1.Toleration) *MockClusterFactory {
- tolerations := factory.get().Spec.Tolerations
+ tolerations := factory.Get().Spec.Tolerations
if len(tolerations) == 0 {
tolerations = []corev1.Toleration{}
}
tolerations = append(tolerations, toleration)
- factory.get().Spec.Tolerations = tolerations
+ factory.Get().Spec.Tolerations = tolerations
return factory
}
@@ -68,21 +63,21 @@ func (factory *MockClusterFactory) AddComponent(compName string, compDefName str
Name: compName,
ComponentDefRef: compDefName,
}
- factory.get().Spec.ComponentSpecs = append(factory.get().Spec.ComponentSpecs, comp)
+ factory.Get().Spec.ComponentSpecs = append(factory.Get().Spec.ComponentSpecs, comp)
return factory
}
func (factory *MockClusterFactory) SetReplicas(replicas int32) *MockClusterFactory {
- comps := factory.get().Spec.ComponentSpecs
+ comps := factory.Get().Spec.ComponentSpecs
if len(comps) > 0 {
comps[len(comps)-1].Replicas = replicas
}
- factory.get().Spec.ComponentSpecs = comps
+ factory.Get().Spec.ComponentSpecs = comps
return factory
}
func (factory *MockClusterFactory) SetServiceAccountName(serviceAccountName string) *MockClusterFactory {
- comps := factory.get().Spec.ComponentSpecs
+ comps := factory.Get().Spec.ComponentSpecs
if len(comps) > 0 {
comps[len(comps)-1].ServiceAccountName = serviceAccountName
}
@@ -90,43 +85,43 @@ func (factory *MockClusterFactory) SetServiceAccountName(serviceAccountName stri
}
func (factory *MockClusterFactory) SetResources(resources corev1.ResourceRequirements) *MockClusterFactory {
- comps := factory.get().Spec.ComponentSpecs
+ comps := factory.Get().Spec.ComponentSpecs
if len(comps) > 0 {
comps[len(comps)-1].Resources = resources
}
- factory.get().Spec.ComponentSpecs = comps
+ factory.Get().Spec.ComponentSpecs = comps
return factory
}
func (factory *MockClusterFactory) SetComponentAffinity(affinity *appsv1alpha1.Affinity) *MockClusterFactory {
- comps := factory.get().Spec.ComponentSpecs
+ comps := factory.Get().Spec.ComponentSpecs
if len(comps) > 0 {
comps[len(comps)-1].Affinity = affinity
}
- factory.get().Spec.ComponentSpecs = comps
+ factory.Get().Spec.ComponentSpecs = comps
return factory
}
func (factory *MockClusterFactory) SetEnabledLogs(logName ...string) *MockClusterFactory {
- comps := factory.get().Spec.ComponentSpecs
+ comps := factory.Get().Spec.ComponentSpecs
if len(comps) > 0 {
comps[len(comps)-1].EnabledLogs = logName
}
- factory.get().Spec.ComponentSpecs = comps
+ factory.Get().Spec.ComponentSpecs = comps
return factory
}
func (factory *MockClusterFactory) SetClassDefRef(classDefRef *appsv1alpha1.ClassDefRef) *MockClusterFactory {
- comps := factory.get().Spec.ComponentSpecs
+ comps := factory.Get().Spec.ComponentSpecs
if len(comps) > 0 {
comps[len(comps)-1].ClassDefRef = classDefRef
}
- factory.get().Spec.ComponentSpecs = comps
+ factory.Get().Spec.ComponentSpecs = comps
return factory
}
func (factory *MockClusterFactory) AddComponentToleration(toleration corev1.Toleration) *MockClusterFactory {
- comps := factory.get().Spec.ComponentSpecs
+ comps := factory.Get().Spec.ComponentSpecs
if len(comps) > 0 {
comp := comps[len(comps)-1]
tolerations := comp.Tolerations
@@ -137,13 +132,13 @@ func (factory *MockClusterFactory) AddComponentToleration(toleration corev1.Tole
comp.Tolerations = tolerations
comps[len(comps)-1] = comp
}
- factory.get().Spec.ComponentSpecs = comps
+ factory.Get().Spec.ComponentSpecs = comps
return factory
}
func (factory *MockClusterFactory) AddVolumeClaimTemplate(volumeName string,
pvcSpec appsv1alpha1.PersistentVolumeClaimSpec) *MockClusterFactory {
- comps := factory.get().Spec.ComponentSpecs
+ comps := factory.Get().Spec.ComponentSpecs
if len(comps) > 0 {
comp := comps[len(comps)-1]
comp.VolumeClaimTemplates = append(comp.VolumeClaimTemplates,
@@ -153,48 +148,48 @@ func (factory *MockClusterFactory) AddVolumeClaimTemplate(volumeName string,
})
comps[len(comps)-1] = comp
}
- factory.get().Spec.ComponentSpecs = comps
+ factory.Get().Spec.ComponentSpecs = comps
return factory
}
func (factory *MockClusterFactory) SetMonitor(monitor bool) *MockClusterFactory {
- comps := factory.get().Spec.ComponentSpecs
+ comps := factory.Get().Spec.ComponentSpecs
if len(comps) > 0 {
comps[len(comps)-1].Monitor = monitor
}
- factory.get().Spec.ComponentSpecs = comps
+ factory.Get().Spec.ComponentSpecs = comps
return factory
}
func (factory *MockClusterFactory) SetSwitchPolicy(switchPolicy *appsv1alpha1.ClusterSwitchPolicy) *MockClusterFactory {
- comps := factory.get().Spec.ComponentSpecs
+ comps := factory.Get().Spec.ComponentSpecs
if len(comps) > 0 {
comps[len(comps)-1].SwitchPolicy = switchPolicy
}
- factory.get().Spec.ComponentSpecs = comps
+ factory.Get().Spec.ComponentSpecs = comps
return factory
}
func (factory *MockClusterFactory) SetTLS(tls bool) *MockClusterFactory {
- comps := factory.get().Spec.ComponentSpecs
+ comps := factory.Get().Spec.ComponentSpecs
if len(comps) > 0 {
comps[len(comps)-1].TLS = tls
}
- factory.get().Spec.ComponentSpecs = comps
+ factory.Get().Spec.ComponentSpecs = comps
return factory
}
func (factory *MockClusterFactory) SetIssuer(issuer *appsv1alpha1.Issuer) *MockClusterFactory {
- comps := factory.get().Spec.ComponentSpecs
+ comps := factory.Get().Spec.ComponentSpecs
if len(comps) > 0 {
comps[len(comps)-1].Issuer = issuer
}
- factory.get().Spec.ComponentSpecs = comps
+ factory.Get().Spec.ComponentSpecs = comps
return factory
}
func (factory *MockClusterFactory) AddService(serviceName string, serviceType corev1.ServiceType) *MockClusterFactory {
- comps := factory.get().Spec.ComponentSpecs
+ comps := factory.Get().Spec.ComponentSpecs
if len(comps) > 0 {
comp := comps[len(comps)-1]
comp.Services = append(comp.Services,
@@ -204,32 +199,20 @@ func (factory *MockClusterFactory) AddService(serviceName string, serviceType co
})
comps[len(comps)-1] = comp
}
- factory.get().Spec.ComponentSpecs = comps
- return factory
-}
-
-func (factory *MockClusterFactory) AddRestorePointInTime(restoreTime metav1.Time, compNames, sourceCluster string) *MockClusterFactory {
- annotations := factory.get().Annotations
- if annotations == nil {
- annotations = map[string]string{}
- }
- annotations[constant.RestoreFromTimeAnnotationKey] = fmt.Sprintf(`{"%s":"%s"}`, compNames, restoreTime.Format(time.RFC3339))
- annotations[constant.RestoreFromSrcClusterAnnotationKey] = sourceCluster
-
- factory.get().Annotations = annotations
+ factory.Get().Spec.ComponentSpecs = comps
return factory
}
func (factory *MockClusterFactory) SetBackup(backup *appsv1alpha1.ClusterBackup) *MockClusterFactory {
- factory.get().Spec.Backup = backup
+ factory.Get().Spec.Backup = backup
return factory
}
func (factory *MockClusterFactory) SetServiceRefs(serviceRefs []appsv1alpha1.ServiceRef) *MockClusterFactory {
- comps := factory.get().Spec.ComponentSpecs
+ comps := factory.Get().Spec.ComponentSpecs
if len(comps) > 0 {
comps[len(comps)-1].ServiceRefs = serviceRefs
}
- factory.get().Spec.ComponentSpecs = comps
+ factory.Get().Spec.ComponentSpecs = comps
return factory
}
diff --git a/internal/testutil/apps/cluster_replication_test_util.go b/internal/testutil/apps/cluster_replication_test_util.go
index 51e5f835191..95e81429595 100644
--- a/internal/testutil/apps/cluster_replication_test_util.go
+++ b/internal/testutil/apps/cluster_replication_test_util.go
@@ -45,7 +45,7 @@ func MockReplicationComponentPod(
SetOwnerReferences("apps/v1", constant.StatefulSetKind, sts).
AddAppInstanceLabel(clusterName).
AddAppComponentLabel(compName).
- AddAppManangedByLabel().
+ AddAppManagedByLabel().
AddRoleLabel(roleName).
AddControllerRevisionHashLabel(sts.Status.UpdateRevision).
AddContainer(corev1.Container{Name: DefaultRedisContainerName, Image: DefaultRedisImageName}).
diff --git a/internal/testutil/apps/cluster_stateless_test_util.go b/internal/testutil/apps/cluster_stateless_test_util.go
index 45f8c0fcefe..69e7ce76476 100644
--- a/internal/testutil/apps/cluster_stateless_test_util.go
+++ b/internal/testutil/apps/cluster_stateless_test_util.go
@@ -50,7 +50,7 @@ func MockStatelessPod(testCtx *testutil.TestContext, deploy *appsv1.Deployment,
SetOwnerReferences("apps/v1", constant.ReplicaSetKind, newRs).
AddAppInstanceLabel(clusterName).
AddAppComponentLabel(componentName).
- AddAppManangedByLabel().
+ AddAppManagedByLabel().
AddContainer(corev1.Container{Name: DefaultNginxContainerName, Image: NginxImage}).
Create(testCtx).GetObject()
}
diff --git a/internal/testutil/apps/clusterdef_factory.go b/internal/testutil/apps/clusterdef_factory.go
index 828c9e71d21..b68e0dcf7cc 100644
--- a/internal/testutil/apps/clusterdef_factory.go
+++ b/internal/testutil/apps/clusterdef_factory.go
@@ -40,7 +40,7 @@ type MockClusterDefFactory struct {
func NewClusterDefFactory(name string) *MockClusterDefFactory {
f := &MockClusterDefFactory{}
- f.init("", name,
+ f.Init("", name,
&appsv1alpha1.ClusterDefinition{
Spec: appsv1alpha1.ClusterDefinitionSpec{
ComponentDefs: []appsv1alpha1.ClusterComponentDefinition{},
@@ -69,7 +69,7 @@ func (factory *MockClusterDefFactory) AddComponentDef(tplType ComponentDefTplTyp
case StatelessNginxComponent:
component = &statelessNginxComponent
}
- factory.get().Spec.ComponentDefs = append(factory.get().Spec.ComponentDefs, *component)
+ factory.Get().Spec.ComponentDefs = append(factory.Get().Spec.ComponentDefs, *component)
comp := factory.getLastCompDef()
comp.Name = compDefName
return factory
@@ -165,24 +165,24 @@ func (factory *MockClusterDefFactory) AddHorizontalScalePolicy(policy appsv1alph
func (factory *MockClusterDefFactory) SetConnectionCredential(
connectionCredential map[string]string, svc *appsv1alpha1.ServiceSpec) *MockClusterDefFactory {
- factory.get().Spec.ConnectionCredential = connectionCredential
+ factory.Get().Spec.ConnectionCredential = connectionCredential
factory.SetServiceSpec(svc)
return factory
}
func (factory *MockClusterDefFactory) get1stCompDef() *appsv1alpha1.ClusterComponentDefinition {
- if len(factory.get().Spec.ComponentDefs) == 0 {
+ if len(factory.Get().Spec.ComponentDefs) == 0 {
return nil
}
- return &factory.get().Spec.ComponentDefs[0]
+ return &factory.Get().Spec.ComponentDefs[0]
}
func (factory *MockClusterDefFactory) getLastCompDef() *appsv1alpha1.ClusterComponentDefinition {
- l := len(factory.get().Spec.ComponentDefs)
+ l := len(factory.Get().Spec.ComponentDefs)
if l == 0 {
return nil
}
- comps := factory.get().Spec.ComponentDefs
+ comps := factory.Get().Spec.ComponentDefs
return &comps[l-1]
}
diff --git a/internal/testutil/apps/clusterversion_factory.go b/internal/testutil/apps/clusterversion_factory.go
index 3ed7d3052d5..f1ed12cb603 100644
--- a/internal/testutil/apps/clusterversion_factory.go
+++ b/internal/testutil/apps/clusterversion_factory.go
@@ -31,7 +31,7 @@ type MockClusterVersionFactory struct {
func NewClusterVersionFactory(name, cdRef string) *MockClusterVersionFactory {
f := &MockClusterVersionFactory{}
- f.init("", name,
+ f.Init("", name,
&appsv1alpha1.ClusterVersion{
Spec: appsv1alpha1.ClusterVersionSpec{
ClusterDefinitionRef: cdRef,
@@ -45,18 +45,18 @@ func (factory *MockClusterVersionFactory) AddComponentVersion(compDefName string
comp := appsv1alpha1.ClusterComponentVersion{
ComponentDefRef: compDefName,
}
- factory.get().Spec.ComponentVersions = append(factory.get().Spec.ComponentVersions, comp)
+ factory.Get().Spec.ComponentVersions = append(factory.Get().Spec.ComponentVersions, comp)
return factory
}
func (factory *MockClusterVersionFactory) AddInitContainer(container corev1.Container) *MockClusterVersionFactory {
- comps := factory.get().Spec.ComponentVersions
+ comps := factory.Get().Spec.ComponentVersions
if len(comps) > 0 {
comp := comps[len(comps)-1]
comp.VersionsCtx.InitContainers = append(comp.VersionsCtx.InitContainers, container)
comps[len(comps)-1] = comp
}
- factory.get().Spec.ComponentVersions = comps
+ factory.Get().Spec.ComponentVersions = comps
return factory
}
@@ -68,13 +68,13 @@ func (factory *MockClusterVersionFactory) AddInitContainerShort(name string, ima
}
func (factory *MockClusterVersionFactory) AddContainer(container corev1.Container) *MockClusterVersionFactory {
- comps := factory.get().Spec.ComponentVersions
+ comps := factory.Get().Spec.ComponentVersions
if len(comps) > 0 {
comp := comps[len(comps)-1]
comp.VersionsCtx.Containers = append(comp.VersionsCtx.Containers, container)
comps[len(comps)-1] = comp
}
- factory.get().Spec.ComponentVersions = comps
+ factory.Get().Spec.ComponentVersions = comps
return factory
}
@@ -87,7 +87,7 @@ func (factory *MockClusterVersionFactory) AddContainerShort(name string, image s
func (factory *MockClusterVersionFactory) AddConfigTemplate(name string,
configTemplateRef string, configConstraintRef string, volumeName string) *MockClusterVersionFactory {
- comps := factory.get().Spec.ComponentVersions
+ comps := factory.Get().Spec.ComponentVersions
if len(comps) > 0 {
comp := comps[len(comps)-1]
comp.ConfigSpecs = append(comp.ConfigSpecs,
@@ -101,6 +101,6 @@ func (factory *MockClusterVersionFactory) AddConfigTemplate(name string,
})
comps[len(comps)-1] = comp
}
- factory.get().Spec.ComponentVersions = comps
+ factory.Get().Spec.ComponentVersions = comps
return factory
}
diff --git a/internal/testutil/apps/common_util.go b/internal/testutil/apps/common_util.go
index 14dbacc4b04..e29d360c111 100644
--- a/internal/testutil/apps/common_util.go
+++ b/internal/testutil/apps/common_util.go
@@ -77,7 +77,7 @@ func ChangeObjStatus[T intctrlutil.Object, PT intctrlutil.PObject[T]](testCtx *t
return testCtx.Cli.Status().Patch(testCtx.Ctx, pobj, patch)
}
-// Helper functions to get object, change its fields in input closure and update it.
+// Helper functions to Get object, change its fields in input closure and update it.
// Each helper is a wrapper of client.Get and client.Patch.
// Each helper returns a Gomega assertion function, which should be passed into
// Eventually() or Consistently() as the first parameter.
@@ -154,7 +154,7 @@ func CheckObj[T intctrlutil.Object, PT intctrlutil.PObject[T]](testCtx *testutil
func List[T intctrlutil.Object, PT intctrlutil.PObject[T],
L intctrlutil.ObjList[T], PL intctrlutil.PObjList[T, L]](
- testCtx *testutil.TestContext, _ func(T, L), opt ...client.ListOption) func(gomega.Gomega) []T {
+ testCtx *testutil.TestContext, _ func(T, PT, L, PL), opt ...client.ListOption) func(gomega.Gomega) []T {
return func(g gomega.Gomega) []T {
var objList L
g.Expect(testCtx.Cli.List(testCtx.Ctx, PL(&objList), opt...)).To(gomega.Succeed())
@@ -290,7 +290,7 @@ func DeleteObject[T intctrlutil.Object, PT intctrlutil.PObject[T]](
// ClearResources clears all resources of the given type T satisfying the input ListOptions.
func ClearResources[T intctrlutil.Object, PT intctrlutil.PObject[T],
L intctrlutil.ObjList[T], PL intctrlutil.PObjList[T, L]](
- testCtx *testutil.TestContext, funcSig func(T, L), opts ...client.DeleteAllOfOption) {
+ testCtx *testutil.TestContext, funcSig func(T, PT, L, PL), opts ...client.DeleteAllOfOption) {
ClearResourcesWithRemoveFinalizerOption[T, PT, L, PL](testCtx, funcSig, false, opts...)
}
@@ -298,7 +298,7 @@ func ClearResources[T intctrlutil.Object, PT intctrlutil.PObject[T],
// removeFinalizer specifier, and satisfying the input ListOptions.
func ClearResourcesWithRemoveFinalizerOption[T intctrlutil.Object, PT intctrlutil.PObject[T],
L intctrlutil.ObjList[T], PL intctrlutil.PObjList[T, L]](
- testCtx *testutil.TestContext, _ func(T, L), removeFinalizer bool, opts ...client.DeleteAllOfOption) {
+ testCtx *testutil.TestContext, _ func(T, PT, L, PL), removeFinalizer bool, opts ...client.DeleteAllOfOption) {
var (
obj T
objList L
diff --git a/internal/testutil/apps/componentclassdefinition_factory.go b/internal/testutil/apps/componentclassdefinition_factory.go
index 1bfdfb322f1..71ab1f4122f 100644
--- a/internal/testutil/apps/componentclassdefinition_factory.go
+++ b/internal/testutil/apps/componentclassdefinition_factory.go
@@ -32,7 +32,7 @@ type MockComponentClassDefinitionFactory struct {
func NewComponentClassDefinitionFactory(name, clusterDefinitionRef, componentType string) *MockComponentClassDefinitionFactory {
f := &MockComponentClassDefinitionFactory{}
- f.init("", name, &appsv1alpha1.ComponentClassDefinition{
+ f.Init("", name, &appsv1alpha1.ComponentClassDefinition{
ObjectMeta: metav1.ObjectMeta{
Name: name,
Labels: map[string]string{
@@ -46,7 +46,7 @@ func NewComponentClassDefinitionFactory(name, clusterDefinitionRef, componentTyp
}
func (factory *MockComponentClassDefinitionFactory) AddClasses(classes []appsv1alpha1.ComponentClass) *MockComponentClassDefinitionFactory {
- groups := factory.get().Spec.Groups
+ groups := factory.Get().Spec.Groups
groups = append(groups, appsv1alpha1.ComponentClassGroup{
Series: []appsv1alpha1.ComponentClassSeries{
{
@@ -54,6 +54,6 @@ func (factory *MockComponentClassDefinitionFactory) AddClasses(classes []appsv1a
},
},
})
- factory.get().Spec.Groups = groups
+ factory.Get().Spec.Groups = groups
return factory
}
diff --git a/internal/testutil/apps/componentresourceconstraint_factory.go b/internal/testutil/apps/componentresourceconstraint_factory.go
index ca5b9498fb5..e3501a3eafa 100644
--- a/internal/testutil/apps/componentresourceconstraint_factory.go
+++ b/internal/testutil/apps/componentresourceconstraint_factory.go
@@ -95,7 +95,7 @@ type MockComponentResourceConstraintFactory struct {
func NewComponentResourceConstraintFactory(name string) *MockComponentResourceConstraintFactory {
f := &MockComponentResourceConstraintFactory{}
- f.init("", name, &appsv1alpha1.ComponentResourceConstraint{
+ f.Init("", name, &appsv1alpha1.ComponentResourceConstraint{
ObjectMeta: metav1.ObjectMeta{
Name: name,
Labels: map[string]string{
@@ -110,7 +110,7 @@ func (factory *MockComponentResourceConstraintFactory) AddConstraints(constraint
var (
tpl string
newConstraints []appsv1alpha1.ResourceConstraintRule
- constraints = factory.get().Spec.Rules
+ constraints = factory.Get().Spec.Rules
)
switch constraintTplType {
case GeneralResourceConstraint:
@@ -124,11 +124,11 @@ func (factory *MockComponentResourceConstraintFactory) AddConstraints(constraint
panic(err)
}
constraints = append(constraints, newConstraints...)
- factory.get().Spec.Rules = constraints
+ factory.Get().Spec.Rules = constraints
return factory
}
func (factory *MockComponentResourceConstraintFactory) AddSelector(selector appsv1alpha1.ClusterResourceConstraintSelector) *MockComponentResourceConstraintFactory {
- factory.get().Spec.Selector = append(factory.get().Spec.Selector, selector)
+ factory.Get().Spec.Selector = append(factory.Get().Spec.Selector, selector)
return factory
}
diff --git a/internal/testutil/apps/constant.go b/internal/testutil/apps/constant.go
index 943105647dc..584f870a740 100644
--- a/internal/testutil/apps/constant.go
+++ b/internal/testutil/apps/constant.go
@@ -54,7 +54,7 @@ const (
DefaultRedisCompSpecName = "redis-rsts"
DefaultRedisImageName = "redis:7.0.5"
DefaultRedisContainerName = "redis"
- DefaultRedisInitContainerName = "redis-init-container"
+ DefaultRedisInitContainerName = "redis-Init-container"
Class1c1gName = "general-1c1g"
Class2c4gName = "general-2c4g"
@@ -271,7 +271,7 @@ var (
Image: DefaultRedisImageName,
ImagePullPolicy: corev1.PullIfNotPresent,
VolumeMounts: defaultReplicationRedisVolumeMounts,
- Command: []string{"/scripts/init.sh"},
+ Command: []string{"/scripts/Init.sh"},
Resources: zeroResRequirements,
}
diff --git a/internal/testutil/apps/deployment_factoy.go b/internal/testutil/apps/deployment_factoy.go
index 75bcdf7e8aa..5d5abcca007 100644
--- a/internal/testutil/apps/deployment_factoy.go
+++ b/internal/testutil/apps/deployment_factoy.go
@@ -33,7 +33,7 @@ type MockDeploymentFactory struct {
func NewDeploymentFactory(namespace, name, clusterName, componentName string) *MockDeploymentFactory {
f := &MockDeploymentFactory{}
- f.init(namespace, name,
+ f.Init(namespace, name,
&appsv1.Deployment{
ObjectMeta: metav1.ObjectMeta{
Labels: map[string]string{
@@ -65,17 +65,17 @@ func NewDeploymentFactory(namespace, name, clusterName, componentName string) *M
}
func (factory *MockDeploymentFactory) SetMinReadySeconds(minReadySeconds int32) *MockDeploymentFactory {
- factory.get().Spec.MinReadySeconds = minReadySeconds
+ factory.Get().Spec.MinReadySeconds = minReadySeconds
return factory
}
func (factory *MockDeploymentFactory) SetReplicas(replicas int32) *MockDeploymentFactory {
- factory.get().Spec.Replicas = &replicas
+ factory.Get().Spec.Replicas = &replicas
return factory
}
func (factory *MockDeploymentFactory) AddVolume(volume corev1.Volume) *MockDeploymentFactory {
- volumes := &factory.get().Spec.Template.Spec.Volumes
+ volumes := &factory.Get().Spec.Template.Spec.Volumes
*volumes = append(*volumes, volume)
return factory
}
@@ -94,7 +94,7 @@ func (factory *MockDeploymentFactory) AddConfigmapVolume(volumeName, configmapNa
}
func (factory *MockDeploymentFactory) AddContainer(container corev1.Container) *MockDeploymentFactory {
- containers := &factory.get().Spec.Template.Spec.Containers
+ containers := &factory.Get().Spec.Template.Spec.Containers
*containers = append(*containers, container)
return factory
}
diff --git a/internal/testutil/apps/pod_factory.go b/internal/testutil/apps/pod_factory.go
index 2054fc4452d..4626a68add1 100644
--- a/internal/testutil/apps/pod_factory.go
+++ b/internal/testutil/apps/pod_factory.go
@@ -29,7 +29,7 @@ type MockPodFactory struct {
func NewPodFactory(namespace, name string) *MockPodFactory {
f := &MockPodFactory{}
- f.init(namespace, name,
+ f.Init(namespace, name,
&corev1.Pod{
Spec: corev1.PodSpec{
Containers: []corev1.Container{},
@@ -39,13 +39,13 @@ func NewPodFactory(namespace, name string) *MockPodFactory {
}
func (factory *MockPodFactory) AddContainer(container corev1.Container) *MockPodFactory {
- containers := &factory.get().Spec.Containers
+ containers := &factory.Get().Spec.Containers
*containers = append(*containers, container)
return factory
}
func (factory *MockPodFactory) AddVolume(volume corev1.Volume) *MockPodFactory {
- volumes := &factory.get().Spec.Volumes
+ volumes := &factory.Get().Spec.Volumes
if volumes == nil {
volumes = &[]corev1.Volume{}
}
@@ -54,6 +54,6 @@ func (factory *MockPodFactory) AddVolume(volume corev1.Volume) *MockPodFactory {
}
func (factory *MockPodFactory) AddNodeName(nodeName string) *MockPodFactory {
- factory.get().Spec.NodeName = nodeName
+ factory.Get().Spec.NodeName = nodeName
return factory
}
diff --git a/internal/testutil/apps/pvc_factoy.go b/internal/testutil/apps/pvc_factoy.go
index 3e32cf06b2b..8016eac4003 100644
--- a/internal/testutil/apps/pvc_factoy.go
+++ b/internal/testutil/apps/pvc_factoy.go
@@ -35,7 +35,7 @@ type MockPersistentVolumeClaimFactory struct {
func NewPersistentVolumeClaimFactory(namespace, name, clusterName, componentName, vctName string) *MockPersistentVolumeClaimFactory {
f := &MockPersistentVolumeClaimFactory{}
volumeMode := corev1.PersistentVolumeFilesystem
- f.init(namespace, name,
+ f.Init(namespace, name,
&corev1.PersistentVolumeClaim{
ObjectMeta: metav1.ObjectMeta{
Labels: map[string]string{
@@ -58,12 +58,12 @@ func NewPersistentVolumeClaimFactory(namespace, name, clusterName, componentName
}
func (factory *MockPersistentVolumeClaimFactory) SetStorageClass(storageClassName string) *MockPersistentVolumeClaimFactory {
- factory.get().Spec.StorageClassName = &storageClassName
+ factory.Get().Spec.StorageClassName = &storageClassName
return factory
}
func (factory *MockPersistentVolumeClaimFactory) SetStorage(storageSize string) *MockPersistentVolumeClaimFactory {
- factory.get().Spec.Resources = corev1.ResourceRequirements{
+ factory.Get().Spec.Resources = corev1.ResourceRequirements{
Requests: corev1.ResourceList{
corev1.ResourceStorage: resource.MustParse(storageSize),
},
@@ -72,6 +72,6 @@ func (factory *MockPersistentVolumeClaimFactory) SetStorage(storageSize string)
}
func (factory *MockPersistentVolumeClaimFactory) SetAnnotations(annotations map[string]string) *MockPersistentVolumeClaimFactory {
- factory.get().Annotations = annotations
+ factory.Get().Annotations = annotations
return factory
}
diff --git a/internal/testutil/apps/restorejob_factory.go b/internal/testutil/apps/restorejob_factory.go
deleted file mode 100644
index 1c46ab9b53c..00000000000
--- a/internal/testutil/apps/restorejob_factory.go
+++ /dev/null
@@ -1,86 +0,0 @@
-/*
-Copyright (C) 2022-2023 ApeCloud Co., Ltd
-
-This file is part of KubeBlocks project
-
-This program is free software: you can redistribute it and/or modify
-it under the terms of the GNU Affero General Public License as published by
-the Free Software Foundation, either version 3 of the License, or
-(at your option) any later version.
-
-This program is distributed in the hope that it will be useful
-but WITHOUT ANY WARRANTY; without even the implied warranty of
-MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
-GNU Affero General Public License for more details.
-
-You should have received a copy of the GNU Affero General Public License
-along with this program. If not, see .
-*/
-
-package apps
-
-import (
- corev1 "k8s.io/api/core/v1"
- metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
-
- dataprotectionv1alpha1 "github.com/apecloud/kubeblocks/apis/dataprotection/v1alpha1"
-)
-
-type MockRestoreJobFactory struct {
- BaseFactory[dataprotectionv1alpha1.RestoreJob, *dataprotectionv1alpha1.RestoreJob, MockRestoreJobFactory]
-}
-
-func NewRestoreJobFactory(namespace, name string) *MockRestoreJobFactory {
- f := &MockRestoreJobFactory{}
- f.init(namespace, name,
- &dataprotectionv1alpha1.RestoreJob{
- Spec: dataprotectionv1alpha1.RestoreJobSpec{
- Target: dataprotectionv1alpha1.TargetCluster{
- LabelsSelector: &metav1.LabelSelector{
- MatchLabels: map[string]string{},
- },
- },
- },
- }, f)
- return f
-}
-
-func (factory *MockRestoreJobFactory) SetBackupJobName(backupJobName string) *MockRestoreJobFactory {
- factory.get().Spec.BackupJobName = backupJobName
- return factory
-}
-
-func (factory *MockRestoreJobFactory) AddTargetMatchLabels(keyAndValues ...string) *MockRestoreJobFactory {
- for k, v := range WithMap(keyAndValues...) {
- factory.get().Spec.Target.LabelsSelector.MatchLabels[k] = v
- }
- return factory
-}
-
-func (factory *MockRestoreJobFactory) SetTargetSecretName(name string) *MockRestoreJobFactory {
- factory.get().Spec.Target.Secret = &dataprotectionv1alpha1.BackupPolicySecret{Name: name}
- return factory
-}
-
-func (factory *MockRestoreJobFactory) AddTargetVolume(volume corev1.Volume) *MockRestoreJobFactory {
- factory.get().Spec.TargetVolumes = append(factory.get().Spec.TargetVolumes, volume)
- return factory
-}
-
-func (factory *MockRestoreJobFactory) AddTargetVolumePVC(volumeName, pvcName string) *MockRestoreJobFactory {
- volume := corev1.Volume{
- Name: volumeName,
- VolumeSource: corev1.VolumeSource{
- PersistentVolumeClaim: &corev1.PersistentVolumeClaimVolumeSource{
- ClaimName: pvcName,
- },
- },
- }
- factory.AddTargetVolume(volume)
- return factory
-}
-
-func (factory *MockRestoreJobFactory) AddTargetVolumeMount(volumeMount corev1.VolumeMount) *MockRestoreJobFactory {
- factory.get().Spec.TargetVolumeMounts = append(factory.get().Spec.TargetVolumeMounts, volumeMount)
- return factory
-}
diff --git a/internal/testutil/apps/rsm_factoy.go b/internal/testutil/apps/rsm_factoy.go
index e7bdee2c6be..6b9e2e11a1f 100644
--- a/internal/testutil/apps/rsm_factoy.go
+++ b/internal/testutil/apps/rsm_factoy.go
@@ -34,7 +34,7 @@ type MockRSMFactory struct {
func NewRSMFactory(namespace, name string, clusterName string, componentName string) *MockRSMFactory {
f := &MockRSMFactory{}
- f.init(namespace, name,
+ f.Init(namespace, name,
&workloads.ReplicatedStateMachine{
ObjectMeta: metav1.ObjectMeta{
Labels: map[string]string{
@@ -69,12 +69,12 @@ func NewRSMFactory(namespace, name string, clusterName string, componentName str
}
func (factory *MockRSMFactory) SetReplicas(replicas int32) *MockRSMFactory {
- factory.get().Spec.Replicas = &replicas
+ factory.Get().Spec.Replicas = &replicas
return factory
}
func (factory *MockRSMFactory) AddVolume(volume corev1.Volume) *MockRSMFactory {
- volumes := &factory.get().Spec.Template.Spec.Volumes
+ volumes := &factory.Get().Spec.Template.Spec.Volumes
*volumes = append(*volumes, volume)
return factory
}
@@ -93,13 +93,13 @@ func (factory *MockRSMFactory) AddConfigmapVolume(volumeName string, configmapNa
}
func (factory *MockRSMFactory) AddVolumeClaimTemplate(pvc corev1.PersistentVolumeClaim) *MockRSMFactory {
- volumeClaimTpls := &factory.get().Spec.VolumeClaimTemplates
+ volumeClaimTpls := &factory.Get().Spec.VolumeClaimTemplates
*volumeClaimTpls = append(*volumeClaimTpls, pvc)
return factory
}
func (factory *MockRSMFactory) AddContainer(container corev1.Container) *MockRSMFactory {
- containers := &factory.get().Spec.Template.Spec.Containers
+ containers := &factory.Get().Spec.Template.Spec.Containers
*containers = append(*containers, container)
return factory
}
diff --git a/internal/testutil/apps/servicedescriptor_factory.go b/internal/testutil/apps/servicedescriptor_factory.go
index da482f4f38e..10bf2f28ea1 100644
--- a/internal/testutil/apps/servicedescriptor_factory.go
+++ b/internal/testutil/apps/servicedescriptor_factory.go
@@ -32,7 +32,7 @@ type MockServiceDescriptorFactory struct {
func NewServiceDescriptorFactory(namespace, name string) *MockServiceDescriptorFactory {
f := &MockServiceDescriptorFactory{}
- f.init(namespace, name,
+ f.Init(namespace, name,
&appsv1alpha1.ServiceDescriptor{
ObjectMeta: metav1.ObjectMeta{
Name: name,
@@ -46,26 +46,26 @@ func NewServiceDescriptorFactory(namespace, name string) *MockServiceDescriptorF
}
func (factory *MockServiceDescriptorFactory) SetServiceKind(serviceKind string) *MockServiceDescriptorFactory {
- factory.get().Spec.ServiceKind = serviceKind
+ factory.Get().Spec.ServiceKind = serviceKind
return factory
}
func (factory *MockServiceDescriptorFactory) SetServiceVersion(serviceVersion string) *MockServiceDescriptorFactory {
- factory.get().Spec.ServiceVersion = serviceVersion
+ factory.Get().Spec.ServiceVersion = serviceVersion
return factory
}
func (factory *MockServiceDescriptorFactory) SetEndpoint(endpoint appsv1alpha1.CredentialVar) *MockServiceDescriptorFactory {
- factory.get().Spec.Endpoint = &endpoint
+ factory.Get().Spec.Endpoint = &endpoint
return factory
}
func (factory *MockServiceDescriptorFactory) SetPort(port appsv1alpha1.CredentialVar) *MockServiceDescriptorFactory {
- factory.get().Spec.Port = &port
+ factory.Get().Spec.Port = &port
return factory
}
func (factory *MockServiceDescriptorFactory) SetAuth(auth appsv1alpha1.ConnectionCredentialAuth) *MockServiceDescriptorFactory {
- factory.get().Spec.Auth = &auth
+ factory.Get().Spec.Auth = &auth
return factory
}
diff --git a/internal/testutil/apps/statefulset_factoy.go b/internal/testutil/apps/statefulset_factoy.go
index 18366814f88..d94e64e0c98 100644
--- a/internal/testutil/apps/statefulset_factoy.go
+++ b/internal/testutil/apps/statefulset_factoy.go
@@ -33,7 +33,7 @@ type MockStatefulSetFactory struct {
func NewStatefulSetFactory(namespace, name string, clusterName string, componentName string) *MockStatefulSetFactory {
f := &MockStatefulSetFactory{}
- f.init(namespace, name,
+ f.Init(namespace, name,
&appsv1.StatefulSet{
ObjectMeta: metav1.ObjectMeta{
Labels: map[string]string{
@@ -68,12 +68,12 @@ func NewStatefulSetFactory(namespace, name string, clusterName string, component
}
func (factory *MockStatefulSetFactory) SetReplicas(replicas int32) *MockStatefulSetFactory {
- factory.get().Spec.Replicas = &replicas
+ factory.Get().Spec.Replicas = &replicas
return factory
}
func (factory *MockStatefulSetFactory) AddVolume(volume corev1.Volume) *MockStatefulSetFactory {
- volumes := &factory.get().Spec.Template.Spec.Volumes
+ volumes := &factory.Get().Spec.Template.Spec.Volumes
*volumes = append(*volumes, volume)
return factory
}
@@ -92,13 +92,13 @@ func (factory *MockStatefulSetFactory) AddConfigmapVolume(volumeName string, con
}
func (factory *MockStatefulSetFactory) AddVolumeClaimTemplate(pvc corev1.PersistentVolumeClaim) *MockStatefulSetFactory {
- volumeClaimTpls := &factory.get().Spec.VolumeClaimTemplates
+ volumeClaimTpls := &factory.Get().Spec.VolumeClaimTemplates
*volumeClaimTpls = append(*volumeClaimTpls, pvc)
return factory
}
func (factory *MockStatefulSetFactory) AddContainer(container corev1.Container) *MockStatefulSetFactory {
- containers := &factory.get().Spec.Template.Spec.Containers
+ containers := &factory.Get().Spec.Template.Spec.Containers
*containers = append(*containers, container)
return factory
}
diff --git a/internal/testutil/dataprotection/backup_factory.go b/internal/testutil/dataprotection/backup_factory.go
new file mode 100644
index 00000000000..8bab1ef48b8
--- /dev/null
+++ b/internal/testutil/dataprotection/backup_factory.go
@@ -0,0 +1,68 @@
+/*
+Copyright (C) 2022-2023 ApeCloud Co., Ltd
+
+This file is part of KubeBlocks project
+
+This program is free software: you can redistribute it and/or modify
+it under the terms of the GNU Affero General Public License as published by
+the Free Software Foundation, either version 3 of the License, or
+(at your option) any later version.
+
+This program is distributed in the hope that it will be useful
+but WITHOUT ANY WARRANTY; without even the implied warranty of
+MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+GNU Affero General Public License for more details.
+
+You should have received a copy of the GNU Affero General Public License
+along with this program. If not, see .
+*/
+
+package dataprotection
+
+import (
+ "time"
+
+ metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
+
+ dpv1alpha1 "github.com/apecloud/kubeblocks/apis/dataprotection/v1alpha1"
+ testapps "github.com/apecloud/kubeblocks/internal/testutil/apps"
+)
+
+type MockBackupFactory struct {
+ testapps.BaseFactory[dpv1alpha1.Backup, *dpv1alpha1.Backup, MockBackupFactory]
+}
+
+func NewBackupFactory(namespace, name string) *MockBackupFactory {
+ f := &MockBackupFactory{}
+ f.Init(namespace, name,
+ &dpv1alpha1.Backup{
+ Spec: dpv1alpha1.BackupSpec{},
+ }, f)
+ return f
+}
+
+func (f *MockBackupFactory) SetBackupPolicyName(backupPolicyName string) *MockBackupFactory {
+ f.Get().Spec.BackupPolicyName = backupPolicyName
+ return f
+}
+
+func (f *MockBackupFactory) SetBackupMethod(backupMethod string) *MockBackupFactory {
+ f.Get().Spec.BackupMethod = backupMethod
+ return f
+}
+
+func (f *MockBackupFactory) SetLabels(labels map[string]string) *MockBackupFactory {
+ f.Get().SetLabels(labels)
+ return f
+}
+
+func (f *MockBackupFactory) SetBackupTimeRange(startTime, stopTime time.Time) *MockBackupFactory {
+ tr := f.Get().Status.TimeRange
+ if tr == nil {
+ tr = &dpv1alpha1.BackupTimeRange{}
+ }
+ tr.Start = &metav1.Time{Time: startTime}
+ tr.End = &metav1.Time{Time: stopTime}
+ f.Get().Status.TimeRange = tr
+ return f
+}
diff --git a/internal/testutil/dataprotection/backup_utils.go b/internal/testutil/dataprotection/backup_utils.go
new file mode 100644
index 00000000000..8dc770b98e3
--- /dev/null
+++ b/internal/testutil/dataprotection/backup_utils.go
@@ -0,0 +1,223 @@
+/*
+Copyright (C) 2022-2023 ApeCloud Co., Ltd
+
+This file is part of KubeBlocks project
+
+This program is free software: you can redistribute it and/or modify
+it under the terms of the GNU Affero General Public License as published by
+the Free Software Foundation, either version 3 of the License, or
+(at your option) any later version.
+
+This program is distributed in the hope that it will be useful
+but WITHOUT ANY WARRANTY; without even the implied warranty of
+MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+GNU Affero General Public License for more details.
+
+You should have received a copy of the GNU Affero General Public License
+along with this program. If not, see .
+*/
+
+package dataprotection
+
+import (
+ . "github.com/onsi/ginkgo/v2"
+ . "github.com/onsi/gomega"
+
+ corev1 "k8s.io/api/core/v1"
+ "sigs.k8s.io/controller-runtime/pkg/client"
+
+ dpv1alpha1 "github.com/apecloud/kubeblocks/apis/dataprotection/v1alpha1"
+ storagev1alpha1 "github.com/apecloud/kubeblocks/apis/storage/v1alpha1"
+ "github.com/apecloud/kubeblocks/internal/constant"
+ "github.com/apecloud/kubeblocks/internal/dataprotection/utils"
+ "github.com/apecloud/kubeblocks/internal/dataprotection/utils/boolptr"
+ "github.com/apecloud/kubeblocks/internal/testutil"
+ testapps "github.com/apecloud/kubeblocks/internal/testutil/apps"
+)
+
+func NewFakeActionSet(testCtx *testutil.TestContext) *dpv1alpha1.ActionSet {
+ as := testapps.CreateCustomizedObj(testCtx, "backup/actionset.yaml",
+ &dpv1alpha1.ActionSet{}, testapps.WithName(ActionSetName))
+ Eventually(testapps.CheckObj(testCtx, client.ObjectKeyFromObject(as),
+ func(g Gomega, as *dpv1alpha1.ActionSet) {
+ g.Expect(as.Status.Phase).Should(BeEquivalentTo(dpv1alpha1.AvailablePhase))
+ })).Should(Succeed())
+ return as
+}
+
+func NewFakeBackupPolicy(testCtx *testutil.TestContext,
+ change func(backupPolicy *dpv1alpha1.BackupPolicy)) *dpv1alpha1.BackupPolicy {
+ bp := NewBackupPolicyFactory(testCtx.DefaultNamespace, BackupPolicyName).
+ SetBackupRepoName(BackupRepoName).
+ SetTarget(constant.AppInstanceLabelKey, ClusterName,
+ constant.KBAppComponentLabelKey, ComponentName,
+ constant.RoleLabelKey, constant.Leader).
+ SetPathPrefix(BackupPathPrefix).
+ SetTargetConnectionCredential(ClusterName).
+ AddBackupMethod(BackupMethodName, false, ActionSetName).
+ SetBackupMethodVolumeMounts(DataVolumeName, DataVolumeMountPath,
+ LogVolumeName, LogVolumeMountPath).
+ AddBackupMethod(VSBackupMethodName, true, "").
+ SetBackupMethodVolumes([]string{DataVolumeName}).
+ Apply(change).
+ Create(testCtx).GetObject()
+ Eventually(testapps.CheckObj(testCtx, client.ObjectKeyFromObject(bp),
+ func(g Gomega, bp *dpv1alpha1.BackupPolicy) {
+ g.Expect(bp.Status.Phase).Should(BeEquivalentTo(dpv1alpha1.AvailablePhase))
+ })).Should(Succeed())
+ return bp
+}
+
+func NewFakeStorageProvider(testCtx *testutil.TestContext,
+ change func(sp *storagev1alpha1.StorageProvider)) *storagev1alpha1.StorageProvider {
+ sp := testapps.CreateCustomizedObj(testCtx, "backup/storageprovider.yaml",
+ &storagev1alpha1.StorageProvider{}, func(obj *storagev1alpha1.StorageProvider) {
+ obj.Name = StorageProviderName
+ if change != nil {
+ change(obj)
+ }
+ })
+ // the storage provider controller is not running, so set the status manually
+ Expect(testapps.ChangeObjStatus(testCtx, sp, func() {
+ sp.Status.Phase = storagev1alpha1.StorageProviderReady
+ })).Should(Succeed())
+ return sp
+}
+
+func NewFakeBackupRepo(testCtx *testutil.TestContext,
+ change func(repo *dpv1alpha1.BackupRepo)) (*dpv1alpha1.BackupRepo, string) {
+ repo := testapps.CreateCustomizedObj(testCtx, "backup/backuprepo.yaml",
+ &dpv1alpha1.BackupRepo{}, func(obj *dpv1alpha1.BackupRepo) {
+ obj.Name = BackupRepoName
+ obj.Spec.StorageProviderRef = StorageProviderName
+ if change != nil {
+ change(obj)
+ }
+ })
+ var name string
+ Eventually(testapps.CheckObj(testCtx, client.ObjectKeyFromObject(repo),
+ func(g Gomega, repo *dpv1alpha1.BackupRepo) {
+ g.Expect(repo.Status.Phase).Should(BeEquivalentTo(dpv1alpha1.BackupRepoReady))
+ g.Expect(repo.Status.BackupPVCName).ShouldNot(BeEmpty())
+ name = repo.Status.BackupPVCName
+ })).Should(Succeed())
+ return repo, name
+}
+
+func NewFakeBackup(testCtx *testutil.TestContext,
+ change func(backup *dpv1alpha1.Backup)) *dpv1alpha1.Backup {
+ if change == nil {
+ change = func(*dpv1alpha1.Backup) {} // set nop
+ }
+ backup := NewBackupFactory(testCtx.DefaultNamespace, BackupName).
+ SetBackupPolicyName(BackupPolicyName).
+ SetBackupMethod(BackupMethodName).
+ Apply(change).
+ Create(testCtx).GetObject()
+ return backup
+}
+
+func NewFakeCluster(testCtx *testutil.TestContext) *BackupClusterInfo {
+ createPVC := func(name string) *corev1.PersistentVolumeClaim {
+ return testapps.NewPersistentVolumeClaimFactory(
+ testCtx.DefaultNamespace, name, ClusterName, ComponentName, "data").
+ SetStorage("1Gi").
+ SetStorageClass(StorageClassName).
+ Create(testCtx).GetObject()
+ }
+
+ podFactory := func(name string) *testapps.MockPodFactory {
+ return testapps.NewPodFactory(testCtx.DefaultNamespace, name).
+ AddAppInstanceLabel(ClusterName).
+ AddAppComponentLabel(ComponentName).
+ AddContainer(corev1.Container{Name: ContainerName, Image: testapps.ApeCloudMySQLImage})
+ }
+
+ By("mocking a cluster")
+ cluster := testapps.NewClusterFactory(testCtx.DefaultNamespace, ClusterName,
+ "test-cd", "test-cv").Create(testCtx).GetObject()
+ podName := ClusterName + "-" + ComponentName
+
+ By("mocking a storage class")
+ _ = testapps.CreateStorageClass(testCtx, StorageClassName, true)
+
+ By("mocking a pvc belonging to the pod 0")
+ pvc := createPVC("data-" + podName + "-0")
+
+ By("mocking a pvc belonging to the pod 1")
+ pvc1 := createPVC("data-" + podName + "-1")
+
+ By("mocking pod 0 belonging to the statefulset")
+ volume := corev1.Volume{Name: DataVolumeName, VolumeSource: corev1.VolumeSource{
+ PersistentVolumeClaim: &corev1.PersistentVolumeClaimVolumeSource{ClaimName: pvc.Name}}}
+ pod := podFactory(podName + "-0").
+ AddRoleLabel("leader").
+ AddVolume(volume).
+ Create(testCtx).GetObject()
+
+ By("mocking pod 1 belonging to the statefulset")
+ volume2 := corev1.Volume{Name: DataVolumeName, VolumeSource: corev1.VolumeSource{
+ PersistentVolumeClaim: &corev1.PersistentVolumeClaimVolumeSource{ClaimName: pvc1.Name}}}
+ _ = podFactory(podName + "-1").
+ AddVolume(volume2).
+ Create(testCtx).GetObject()
+
+ return &BackupClusterInfo{
+ Cluster: cluster,
+ TargetPod: pod,
+ TargetPVC: pvc.Name,
+ }
+}
+
+func NewFakeBackupSchedule(testCtx *testutil.TestContext,
+ change func(schedule *dpv1alpha1.BackupSchedule)) *dpv1alpha1.BackupSchedule {
+ schedule := NewBackupScheduleFactory(testCtx.DefaultNamespace, BackupScheduleName).
+ SetBackupPolicyName(BackupPolicyName).
+ SetStartingDeadlineMinutes(StartingDeadlineMinutes).
+ AddSchedulePolicy(dpv1alpha1.SchedulePolicy{
+ Enabled: boolptr.False(),
+ BackupMethod: BackupMethodName,
+ CronExpression: BackupScheduleCron,
+ RetentionPeriod: BackupRetention,
+ }).
+ AddSchedulePolicy(dpv1alpha1.SchedulePolicy{
+ Enabled: boolptr.False(),
+ BackupMethod: VSBackupMethodName,
+ CronExpression: BackupScheduleCron,
+ RetentionPeriod: BackupRetention,
+ }).
+ Apply(change).
+ Create(testCtx).GetObject()
+ return schedule
+}
+
+// EnableBackupSchedule enables the backup schedule that matches the given method.
+func EnableBackupSchedule(testCtx *testutil.TestContext,
+ backupSchedule *dpv1alpha1.BackupSchedule, method string) {
+ Eventually(testapps.ChangeObj(testCtx, backupSchedule, func(schedule *dpv1alpha1.BackupSchedule) {
+ for i := range schedule.Spec.Schedules {
+ if schedule.Spec.Schedules[i].BackupMethod == method {
+ schedule.Spec.Schedules[i].Enabled = boolptr.True()
+ break
+ }
+ }
+ })).Should(Succeed())
+}
+
+func MockBackupStatusMethod(backup *dpv1alpha1.Backup, targetVolume string) {
+ snapshot := utils.VolumeSnapshotEnabled()
+ backupMethod := BackupMethodName
+ if snapshot {
+ backupMethod = VSBackupMethodName
+ }
+ backup.Status.BackupMethod = &dpv1alpha1.BackupMethod{
+ Name: backupMethod,
+ SnapshotVolumes: &snapshot,
+ TargetVolumes: &dpv1alpha1.TargetVolumeInfo{
+ Volumes: []string{targetVolume},
+ VolumeMounts: []corev1.VolumeMount{
+ {Name: targetVolume, MountPath: "/"},
+ },
+ },
+ }
+}
diff --git a/internal/testutil/dataprotection/backuppolicy_factory.go b/internal/testutil/dataprotection/backuppolicy_factory.go
new file mode 100644
index 00000000000..f97b2e8a6bc
--- /dev/null
+++ b/internal/testutil/dataprotection/backuppolicy_factory.go
@@ -0,0 +1,106 @@
+/*
+Copyright (C) 2022-2023 ApeCloud Co., Ltd
+
+This file is part of KubeBlocks project
+
+This program is free software: you can redistribute it and/or modify
+it under the terms of the GNU Affero General Public License as published by
+the Free Software Foundation, either version 3 of the License, or
+(at your option) any later version.
+
+This program is distributed in the hope that it will be useful
+but WITHOUT ANY WARRANTY; without even the implied warranty of
+MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+GNU Affero General Public License for more details.
+
+You should have received a copy of the GNU Affero General Public License
+along with this program. If not, see .
+*/
+
+package dataprotection
+
+import (
+ corev1 "k8s.io/api/core/v1"
+ metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
+
+ dpv1alpha1 "github.com/apecloud/kubeblocks/apis/dataprotection/v1alpha1"
+ testapps "github.com/apecloud/kubeblocks/internal/testutil/apps"
+)
+
+type MockBackupPolicyFactory struct {
+ testapps.BaseFactory[dpv1alpha1.BackupPolicy, *dpv1alpha1.BackupPolicy, MockBackupPolicyFactory]
+}
+
+func NewBackupPolicyFactory(namespace, name string) *MockBackupPolicyFactory {
+ f := &MockBackupPolicyFactory{}
+ f.Init(namespace, name, &dpv1alpha1.BackupPolicy{}, f)
+ return f
+}
+
+func (f *MockBackupPolicyFactory) SetBackupRepoName(backupRepoName string) *MockBackupPolicyFactory {
+ if backupRepoName == "" {
+ f.Get().Spec.BackupRepoName = nil
+ } else {
+ f.Get().Spec.BackupRepoName = &backupRepoName
+ }
+ return f
+}
+
+func (f *MockBackupPolicyFactory) SetPathPrefix(pathPrefix string) *MockBackupPolicyFactory {
+ f.Get().Spec.PathPrefix = pathPrefix
+ return f
+}
+
+func (f *MockBackupPolicyFactory) SetBackoffLimit(backoffLimit int32) *MockBackupPolicyFactory {
+ f.Get().Spec.BackoffLimit = &backoffLimit
+ return f
+}
+
+func (f *MockBackupPolicyFactory) AddBackupMethod(name string,
+ snapshotVolumes bool, actionSetName string) *MockBackupPolicyFactory {
+ f.Get().Spec.BackupMethods = append(f.Get().Spec.BackupMethods,
+ dpv1alpha1.BackupMethod{
+ Name: name,
+ SnapshotVolumes: &snapshotVolumes,
+ ActionSetName: actionSetName,
+ TargetVolumes: &dpv1alpha1.TargetVolumeInfo{},
+ })
+ return f
+}
+
+func (f *MockBackupPolicyFactory) SetBackupMethodVolumes(names []string) *MockBackupPolicyFactory {
+ f.Get().Spec.BackupMethods[len(f.Get().Spec.BackupMethods)-1].TargetVolumes.Volumes = names
+ return f
+}
+
+func (f *MockBackupPolicyFactory) SetBackupMethodVolumeMounts(keyAndValues ...string) *MockBackupPolicyFactory {
+ var volumeMounts []corev1.VolumeMount
+ for k, v := range testapps.WithMap(keyAndValues...) {
+ volumeMounts = append(volumeMounts, corev1.VolumeMount{
+ Name: k,
+ MountPath: v,
+ })
+ }
+ f.Get().Spec.BackupMethods[len(f.Get().Spec.BackupMethods)-1].TargetVolumes.VolumeMounts = volumeMounts
+ return f
+}
+
+func (f *MockBackupPolicyFactory) SetTarget(keyAndValues ...string) *MockBackupPolicyFactory {
+ f.Get().Spec.Target = &dpv1alpha1.BackupTarget{
+ PodSelector: &dpv1alpha1.PodSelector{
+ LabelSelector: &metav1.LabelSelector{
+ MatchLabels: testapps.WithMap(keyAndValues...),
+ },
+ },
+ }
+ return f
+}
+
+func (f *MockBackupPolicyFactory) SetTargetConnectionCredential(secretName string) *MockBackupPolicyFactory {
+ f.Get().Spec.Target.ConnectionCredential = &dpv1alpha1.ConnectionCredential{
+ SecretName: secretName,
+ UsernameKey: "username",
+ PasswordKey: "password",
+ }
+ return f
+}
diff --git a/internal/testutil/apps/backuprepo_factory.go b/internal/testutil/dataprotection/backuprepo_factory.go
similarity index 69%
rename from internal/testutil/apps/backuprepo_factory.go
rename to internal/testutil/dataprotection/backuprepo_factory.go
index 4e4b1e885f4..48a9b74c292 100644
--- a/internal/testutil/apps/backuprepo_factory.go
+++ b/internal/testutil/dataprotection/backuprepo_factory.go
@@ -17,25 +17,26 @@ You should have received a copy of the GNU Affero General Public License
along with this program. If not, see .
*/
-package apps
+package dataprotection
import (
corev1 "k8s.io/api/core/v1"
"k8s.io/apimachinery/pkg/api/resource"
- dataprotectionv1alpha1 "github.com/apecloud/kubeblocks/apis/dataprotection/v1alpha1"
- "github.com/apecloud/kubeblocks/internal/constant"
+ dpv1alpha1 "github.com/apecloud/kubeblocks/apis/dataprotection/v1alpha1"
+ dptypes "github.com/apecloud/kubeblocks/internal/dataprotection/types"
+ testapps "github.com/apecloud/kubeblocks/internal/testutil/apps"
)
type MockBackupRepoFactory struct {
- BaseFactory[dataprotectionv1alpha1.BackupRepo, *dataprotectionv1alpha1.BackupRepo, MockBackupRepoFactory]
+ testapps.BaseFactory[dpv1alpha1.BackupRepo, *dpv1alpha1.BackupRepo, MockBackupRepoFactory]
}
func NewBackupRepoFactory(namespace, name string) *MockBackupRepoFactory {
f := &MockBackupRepoFactory{}
- f.init(namespace, name,
- &dataprotectionv1alpha1.BackupRepo{
- Spec: dataprotectionv1alpha1.BackupRepoSpec{
+ f.Init(namespace, name,
+ &dpv1alpha1.BackupRepo{
+ Spec: dpv1alpha1.BackupRepoSpec{
VolumeCapacity: resource.MustParse("100Gi"),
PVReclaimPolicy: "Retain",
},
@@ -44,39 +45,39 @@ func NewBackupRepoFactory(namespace, name string) *MockBackupRepoFactory {
}
func (factory *MockBackupRepoFactory) SetStorageProviderRef(providerName string) *MockBackupRepoFactory {
- factory.get().Spec.StorageProviderRef = providerName
+ factory.Get().Spec.StorageProviderRef = providerName
return factory
}
func (factory *MockBackupRepoFactory) SetVolumeCapacity(amount string) *MockBackupRepoFactory {
- factory.get().Spec.VolumeCapacity = resource.MustParse(amount)
+ factory.Get().Spec.VolumeCapacity = resource.MustParse(amount)
return factory
}
func (factory *MockBackupRepoFactory) SetPVReclaimPolicy(policy string) *MockBackupRepoFactory {
- factory.get().Spec.PVReclaimPolicy = corev1.PersistentVolumeReclaimPolicy(policy)
+ factory.Get().Spec.PVReclaimPolicy = corev1.PersistentVolumeReclaimPolicy(policy)
return factory
}
func (factory *MockBackupRepoFactory) SetConfig(config map[string]string) *MockBackupRepoFactory {
- factory.get().Spec.Config = config
+ factory.Get().Spec.Config = config
return factory
}
func (factory *MockBackupRepoFactory) SetCredential(ref *corev1.SecretReference) *MockBackupRepoFactory {
- factory.get().Spec.Credential = ref
+ factory.Get().Spec.Credential = ref
return factory
}
func (factory *MockBackupRepoFactory) SetAsDefaultRepo(v bool) *MockBackupRepoFactory {
if v {
- obj := factory.get()
+ obj := factory.Get()
if obj.Annotations == nil {
obj.Annotations = map[string]string{}
}
- obj.Annotations[constant.DefaultBackupRepoAnnotationKey] = "true"
+ obj.Annotations[dptypes.DefaultBackupRepoAnnotationKey] = "true"
} else {
- delete(factory.get().Annotations, constant.DefaultBackupRepoAnnotationKey)
+ delete(factory.Get().Annotations, dptypes.DefaultBackupRepoAnnotationKey)
}
return factory
}
diff --git a/internal/testutil/dataprotection/backupschedule_factory.go b/internal/testutil/dataprotection/backupschedule_factory.go
new file mode 100644
index 00000000000..ad59caf1946
--- /dev/null
+++ b/internal/testutil/dataprotection/backupschedule_factory.go
@@ -0,0 +1,56 @@
+/*
+Copyright (C) 2022-2023 ApeCloud Co., Ltd
+
+This file is part of KubeBlocks project
+
+This program is free software: you can redistribute it and/or modify
+it under the terms of the GNU Affero General Public License as published by
+the Free Software Foundation, either version 3 of the License, or
+(at your option) any later version.
+
+This program is distributed in the hope that it will be useful
+but WITHOUT ANY WARRANTY; without even the implied warranty of
+MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+GNU Affero General Public License for more details.
+
+You should have received a copy of the GNU Affero General Public License
+along with this program. If not, see .
+*/
+
+package dataprotection
+
+import (
+ dpv1alpha1 "github.com/apecloud/kubeblocks/apis/dataprotection/v1alpha1"
+ testapps "github.com/apecloud/kubeblocks/internal/testutil/apps"
+)
+
+type BackupScheduleFactory struct {
+ testapps.BaseFactory[dpv1alpha1.BackupSchedule, *dpv1alpha1.BackupSchedule, BackupScheduleFactory]
+}
+
+func NewBackupScheduleFactory(namespace, name string) *BackupScheduleFactory {
+ f := &BackupScheduleFactory{}
+ f.Init(namespace, name, &dpv1alpha1.BackupSchedule{}, f)
+ f.Get().Spec.Schedules = []dpv1alpha1.SchedulePolicy{}
+ return f
+}
+
+func (f *BackupScheduleFactory) SetBackupPolicyName(backupPolicyName string) *BackupScheduleFactory {
+ f.Get().Spec.BackupPolicyName = backupPolicyName
+ return f
+}
+
+func (f *BackupScheduleFactory) SetStartingDeadlineMinutes(minutes int64) *BackupScheduleFactory {
+ f.Get().Spec.StartingDeadlineMinutes = &minutes
+ return f
+}
+
+func (f *BackupScheduleFactory) AddSchedulePolicy(schedulePolicy dpv1alpha1.SchedulePolicy) *BackupScheduleFactory {
+ f.Get().Spec.Schedules = append(f.Get().Spec.Schedules, schedulePolicy)
+ return f
+}
+
+func (f *BackupScheduleFactory) SetSchedules(schedules []dpv1alpha1.SchedulePolicy) *BackupScheduleFactory {
+ f.Get().Spec.Schedules = schedules
+ return f
+}
diff --git a/internal/testutil/dataprotection/constant.go b/internal/testutil/dataprotection/constant.go
new file mode 100644
index 00000000000..e873292ef08
--- /dev/null
+++ b/internal/testutil/dataprotection/constant.go
@@ -0,0 +1,50 @@
+/*
+Copyright (C) 2022-2023 ApeCloud Co., Ltd
+
+This file is part of KubeBlocks project
+
+This program is free software: you can redistribute it and/or modify
+it under the terms of the GNU Affero General Public License as published by
+the Free Software Foundation, either version 3 of the License, or
+(at your option) any later version.
+
+This program is distributed in the hope that it will be useful
+but WITHOUT ANY WARRANTY; without even the implied warranty of
+MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+GNU Affero General Public License for more details.
+
+You should have received a copy of the GNU Affero General Public License
+along with this program. If not, see .
+*/
+
+package dataprotection
+
+const (
+ ClusterName = "test-cluster"
+ ComponentName = "test-comp"
+ ContainerName = "test-container"
+
+ BackupName = "test-backup"
+ BackupRepoName = "test-repo"
+ BackupPolicyName = "test-backup-policy"
+ BackupMethodName = "xtrabackup"
+ VSBackupMethodName = "volume-snapshot"
+ BackupPathPrefix = "/backup"
+ ActionSetName = "xtrabackup"
+ VSActionSetName = "volume-snapshot"
+
+ DataVolumeName = "data"
+ DataVolumeMountPath = "/data"
+ LogVolumeName = "log"
+ LogVolumeMountPath = "/log"
+
+ StorageProviderName = "test-sp"
+ StorageClassName = "test-sc"
+
+ BackupScheduleName = "test-backup-schedule"
+ BackupScheduleCron = "0 3 * * *"
+ BackupRetention = "7d"
+ StartingDeadlineMinutes = 10
+
+ KBToolImage = "apecloud/kubeblocks-tool:latest"
+)
diff --git a/internal/testutil/dataprotection/k8s_utils.go b/internal/testutil/dataprotection/k8s_utils.go
new file mode 100644
index 00000000000..c332d894929
--- /dev/null
+++ b/internal/testutil/dataprotection/k8s_utils.go
@@ -0,0 +1,33 @@
+/*
+Copyright (C) 2022-2023 ApeCloud Co., Ltd
+
+This file is part of KubeBlocks project
+
+This program is free software: you can redistribute it and/or modify
+it under the terms of the GNU Affero General Public License as published by
+the Free Software Foundation, either version 3 of the License, or
+(at your option) any later version.
+
+This program is distributed in the hope that it will be useful
+but WITHOUT ANY WARRANTY; without even the implied warranty of
+MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+GNU Affero General Public License for more details.
+
+You should have received a copy of the GNU Affero General Public License
+along with this program. If not, see .
+*/
+
+package dataprotection
+
+import (
+ corev1 "k8s.io/api/core/v1"
+
+ "github.com/apecloud/kubeblocks/internal/testutil"
+ testapps "github.com/apecloud/kubeblocks/internal/testutil/apps"
+)
+
+func NewFakePVC(testCtx *testutil.TestContext, name string) *corev1.PersistentVolumeClaim {
+ return testapps.NewPersistentVolumeClaimFactory(testCtx.DefaultNamespace, name, "", "", "").
+ SetStorage("1Gi").
+ Create(testCtx).GetObject()
+}
diff --git a/internal/testutil/dataprotection/restore_factory.go b/internal/testutil/dataprotection/restore_factory.go
new file mode 100644
index 00000000000..8e8f3048f9f
--- /dev/null
+++ b/internal/testutil/dataprotection/restore_factory.go
@@ -0,0 +1,86 @@
+/*
+Copyright (C) 2022-2023 ApeCloud Co., Ltd
+
+This file is part of KubeBlocks project
+
+This program is free software: you can redistribute it and/or modify
+it under the terms of the GNU Affero General Public License as published by
+the Free Software Foundation, either version 3 of the License, or
+(at your option) any later version.
+
+This program is distributed in the hope that it will be useful
+but WITHOUT ANY WARRANTY; without even the implied warranty of
+MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+GNU Affero General Public License for more details.
+
+You should have received a copy of the GNU Affero General Public License
+along with this program. If not, see .
+*/
+
+package dataprotection
+
+import (
+ corev1 "k8s.io/api/core/v1"
+
+ dpv1alpha1 "github.com/apecloud/kubeblocks/apis/dataprotection/v1alpha1"
+ testapps "github.com/apecloud/kubeblocks/internal/testutil/apps"
+)
+
+type MockRestoreFactory struct {
+ testapps.BaseFactory[dpv1alpha1.Restore, *dpv1alpha1.Restore, MockRestoreFactory]
+}
+
+func NewRestoreJobFactory(namespace, name string) *MockRestoreFactory {
+ f := &MockRestoreFactory{}
+ // f.init(namespace, name,
+ // &dpv1alpha1.RestoreJob{
+ // Spec: dpv1alpha1.RestoreJobSpec{
+ // Target: dpv1alpha1.TargetCluster{
+ // LabelsSelector: &metav1.LabelSelector{
+ // MatchLabels: map[string]string{},
+ // },
+ // },
+ // },
+ // }, f)
+ return f
+}
+
+func (factory *MockRestoreFactory) SetBackupName(backupName string) *MockRestoreFactory {
+ // factory.get().Spec.Backup.Name = backupName
+ return factory
+}
+
+func (factory *MockRestoreFactory) AddTargetMatchLabels(keyAndValues ...string) *MockRestoreFactory {
+ // for k, v := range WithMap(keyAndValues...) {
+ // factory.get().Spec.Target.LabelsSelector.MatchLabels[k] = v
+ // }
+ return factory
+}
+
+func (factory *MockRestoreFactory) SetTargetSecretName(name string) *MockRestoreFactory {
+ // factory.get().Spec.Target.Secret = &dataprotectionv1alpha1.BackupPolicySecret{Name: name}
+ return factory
+}
+
+func (factory *MockRestoreFactory) AddTargetVolume(volume corev1.Volume) *MockRestoreFactory {
+ // factory.get().Spec.TargetVolumes = append(factory.get().Spec.TargetVolumes, volume)
+ return factory
+}
+
+func (factory *MockRestoreFactory) AddTargetVolumePVC(volumeName, pvcName string) *MockRestoreFactory {
+ volume := corev1.Volume{
+ Name: volumeName,
+ VolumeSource: corev1.VolumeSource{
+ PersistentVolumeClaim: &corev1.PersistentVolumeClaimVolumeSource{
+ ClaimName: pvcName,
+ },
+ },
+ }
+ factory.AddTargetVolume(volume)
+ return factory
+}
+
+func (factory *MockRestoreFactory) AddTargetVolumeMount(volumeMount corev1.VolumeMount) *MockRestoreFactory {
+ // factory.get().Spec.TargetVolumeMounts = append(factory.get().Spec.TargetVolumeMounts, volumeMount)
+ return factory
+}
diff --git a/controllers/apps/components/stateful_workload.go b/internal/testutil/dataprotection/types.go
similarity index 70%
rename from controllers/apps/components/stateful_workload.go
rename to internal/testutil/dataprotection/types.go
index acf6819a453..3c249ea1c85 100644
--- a/controllers/apps/components/stateful_workload.go
+++ b/internal/testutil/dataprotection/types.go
@@ -17,14 +17,16 @@ You should have received a copy of the GNU Affero General Public License
along with this program. If not, see .
*/
-package components
+package dataprotection
-type statefulComponentWorkloadBuilder struct {
- componentWorkloadBuilderBase
-}
+import (
+ corev1 "k8s.io/api/core/v1"
-var _ componentWorkloadBuilder = &statefulComponentWorkloadBuilder{}
+ appsv1alpha1 "github.com/apecloud/kubeblocks/apis/apps/v1alpha1"
+)
-func (b *statefulComponentWorkloadBuilder) BuildWorkload() componentWorkloadBuilder {
- return b.BuildWorkload4StatefulSet("stateful")
+type BackupClusterInfo struct {
+ Cluster *appsv1alpha1.Cluster
+ TargetPod *corev1.Pod
+ TargetPVC string
}
diff --git a/internal/testutil/dataprotection/utils.go b/internal/testutil/dataprotection/utils.go
new file mode 100644
index 00000000000..368bd34a9fe
--- /dev/null
+++ b/internal/testutil/dataprotection/utils.go
@@ -0,0 +1,60 @@
+/*
+Copyright (C) 2022-2023 ApeCloud Co., Ltd
+
+This file is part of KubeBlocks project
+
+This program is free software: you can redistribute it and/or modify
+it under the terms of the GNU Affero General Public License as published by
+the Free Software Foundation, either version 3 of the License, or
+(at your option) any later version.
+
+This program is distributed in the hope that it will be useful
+but WITHOUT ANY WARRANTY; without even the implied warranty of
+MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+GNU Affero General Public License for more details.
+
+You should have received a copy of the GNU Affero General Public License
+along with this program. If not, see .
+*/
+
+package dataprotection
+
+import (
+ . "github.com/onsi/gomega"
+
+ vsv1 "github.com/kubernetes-csi/external-snapshotter/client/v6/apis/volumesnapshot/v1"
+ batchv1 "k8s.io/api/batch/v1"
+ corev1 "k8s.io/api/core/v1"
+ "sigs.k8s.io/controller-runtime/pkg/client"
+
+ dpv1alpha1 "github.com/apecloud/kubeblocks/apis/dataprotection/v1alpha1"
+ "github.com/apecloud/kubeblocks/internal/testutil"
+ testapps "github.com/apecloud/kubeblocks/internal/testutil/apps"
+)
+
+func PatchK8sJobStatus(testCtx *testutil.TestContext, key client.ObjectKey, jobStatus batchv1.JobConditionType) {
+ Eventually(testapps.GetAndChangeObjStatus(testCtx, key, func(fetched *batchv1.Job) {
+ jobCondition := batchv1.JobCondition{Type: jobStatus, Status: corev1.ConditionTrue}
+ fetched.Status.Conditions = append(fetched.Status.Conditions, jobCondition)
+ })).Should(Succeed())
+}
+
+func ReplaceK8sJobStatus(testCtx *testutil.TestContext, key client.ObjectKey, jobStatus batchv1.JobConditionType) {
+ Eventually(testapps.GetAndChangeObjStatus(testCtx, key, func(fetched *batchv1.Job) {
+ jobCondition := batchv1.JobCondition{Type: jobStatus, Status: corev1.ConditionTrue}
+ fetched.Status.Conditions = []batchv1.JobCondition{jobCondition}
+ })).Should(Succeed())
+}
+
+func PatchVolumeSnapshotStatus(testCtx *testutil.TestContext, key client.ObjectKey, readyToUse bool) {
+ Eventually(testapps.GetAndChangeObjStatus(testCtx, key, func(fetched *vsv1.VolumeSnapshot) {
+ snapStatus := vsv1.VolumeSnapshotStatus{ReadyToUse: &readyToUse}
+ fetched.Status = &snapStatus
+ })).Should(Succeed())
+}
+
+func PatchBackupStatus(testCtx *testutil.TestContext, key client.ObjectKey, status dpv1alpha1.BackupStatus) {
+ Eventually(testapps.GetAndChangeObjStatus(testCtx, key, func(fetched *dpv1alpha1.Backup) {
+ fetched.Status = status
+ })).Should(Succeed())
+}
diff --git a/internal/testutil/dataprotection/vs_factory.go b/internal/testutil/dataprotection/vs_factory.go
new file mode 100644
index 00000000000..4f5aa3c071e
--- /dev/null
+++ b/internal/testutil/dataprotection/vs_factory.go
@@ -0,0 +1,44 @@
+/*
+Copyright (C) 2022-2023 ApeCloud Co., Ltd
+
+This file is part of KubeBlocks project
+
+This program is free software: you can redistribute it and/or modify
+it under the terms of the GNU Affero General Public License as published by
+the Free Software Foundation, either version 3 of the License, or
+(at your option) any later version.
+
+This program is distributed in the hope that it will be useful
+but WITHOUT ANY WARRANTY; without even the implied warranty of
+MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+GNU Affero General Public License for more details.
+
+You should have received a copy of the GNU Affero General Public License
+along with this program. If not, see .
+*/
+
+package dataprotection
+
+import (
+ vsv1 "github.com/kubernetes-csi/external-snapshotter/client/v6/apis/volumesnapshot/v1"
+
+ testapps "github.com/apecloud/kubeblocks/internal/testutil/apps"
+)
+
+type MockVolumeSnapshotFactory struct {
+ testapps.BaseFactory[vsv1.VolumeSnapshot, *vsv1.VolumeSnapshot, MockVolumeSnapshotFactory]
+}
+
+func NewVolumeSnapshotFactory(namespace, name string) *MockVolumeSnapshotFactory {
+ f := &MockVolumeSnapshotFactory{}
+ f.Init(namespace, name,
+ &vsv1.VolumeSnapshot{
+ Spec: vsv1.VolumeSnapshotSpec{},
+ }, f)
+ return f
+}
+
+func (f *MockVolumeSnapshotFactory) SetSourcePVCName(name string) *MockVolumeSnapshotFactory {
+ f.Get().Spec.Source.PersistentVolumeClaimName = &name
+ return f
+}
diff --git a/internal/testutil/k8s/storage_util.go b/internal/testutil/k8s/storage_util.go
index 77f0fb01343..53b01248b7a 100644
--- a/internal/testutil/k8s/storage_util.go
+++ b/internal/testutil/k8s/storage_util.go
@@ -20,13 +20,23 @@ along with this program. If not, see .
package testutil
import (
+ snapshotv1 "github.com/kubernetes-csi/external-snapshotter/client/v6/apis/volumesnapshot/v1"
"github.com/onsi/gomega"
storagev1 "k8s.io/api/storage/v1"
+ metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
+ "k8s.io/apimachinery/pkg/types"
"k8s.io/kubectl/pkg/util/storage"
+ "sigs.k8s.io/controller-runtime/pkg/client"
"github.com/apecloud/kubeblocks/internal/testutil"
)
+var (
+ DefaultStorageClassName string = "default-sc-for-testing"
+ defaultVolumeSnapshotClassName string = "default-vsc-for-testing"
+ defaultProvisioner string = "testing.kubeblocks.io"
+)
+
func GetDefaultStorageClass(testCtx *testutil.TestContext) *storagev1.StorageClass {
scList := &storagev1.StorageClassList{}
gomega.Expect(testCtx.Cli.List(testCtx.Ctx, scList)).Should(gomega.Succeed())
@@ -49,3 +59,100 @@ func GetDefaultStorageClass(testCtx *testutil.TestContext) *storagev1.StorageCla
func isDefaultStorageClassAnnotation(storageClass *storagev1.StorageClass) bool {
return storageClass.Annotations != nil && storageClass.Annotations[storage.IsDefaultStorageClassAnnotation] == "true"
}
+
+func CreateMockStorageClass(testCtx *testutil.TestContext, storageClassName string) *storagev1.StorageClass {
+ sc := getStorageClass(testCtx, storageClassName)
+ if sc == nil {
+ sc = createStorageClass(testCtx, storageClassName)
+ }
+ return sc
+}
+
+func MockEnableVolumeSnapshot(testCtx *testutil.TestContext, storageClassName string) {
+ sc := getStorageClass(testCtx, storageClassName)
+ if sc == nil {
+ sc = createStorageClass(testCtx, storageClassName)
+ }
+ vsc := getVolumeSnapshotClass(testCtx, sc)
+ if vsc == nil {
+ createVolumeSnapshotClass(testCtx, sc)
+ }
+ gomega.Expect(IsMockVolumeSnapshotEnabled(testCtx, storageClassName)).Should(gomega.BeTrue())
+}
+
+func MockDisableVolumeSnapshot(testCtx *testutil.TestContext, storageClassName string) {
+ sc := getStorageClass(testCtx, storageClassName)
+ if sc != nil {
+ deleteVolumeSnapshotClass(testCtx, sc)
+ deleteStorageClass(testCtx, storageClassName)
+ }
+}
+
+func IsMockVolumeSnapshotEnabled(testCtx *testutil.TestContext, storageClassName string) bool {
+ sc := getStorageClass(testCtx, storageClassName)
+ if sc == nil {
+ return false
+ }
+ return getVolumeSnapshotClass(testCtx, sc) != nil
+}
+
+func getStorageClass(testCtx *testutil.TestContext, storageClassName string) *storagev1.StorageClass {
+ sc := &storagev1.StorageClass{}
+ if err := testCtx.Cli.Get(testCtx.Ctx, types.NamespacedName{Name: storageClassName}, sc); err != nil {
+ if client.IgnoreNotFound(err) == nil {
+ return nil
+ }
+ gomega.Expect(err).Should(gomega.Succeed())
+ }
+ return sc
+}
+
+func createStorageClass(testCtx *testutil.TestContext, storageClassName string) *storagev1.StorageClass {
+ allowVolumeExpansion := true
+ sc := &storagev1.StorageClass{
+ ObjectMeta: metav1.ObjectMeta{
+ Name: storageClassName,
+ },
+ Provisioner: defaultProvisioner,
+ AllowVolumeExpansion: &allowVolumeExpansion,
+ }
+ gomega.Expect(testCtx.Cli.Create(testCtx.Ctx, sc)).Should(gomega.Succeed())
+ return sc
+}
+
+func deleteStorageClass(testCtx *testutil.TestContext, storageClassName string) {
+ sc := getStorageClass(testCtx, storageClassName)
+ if sc != nil {
+ gomega.Expect(testCtx.Cli.Delete(testCtx.Ctx, sc)).Should(gomega.Succeed())
+ }
+}
+
+func getVolumeSnapshotClass(testCtx *testutil.TestContext, storageClass *storagev1.StorageClass) *snapshotv1.VolumeSnapshotClass {
+ vscList := &snapshotv1.VolumeSnapshotClassList{}
+ gomega.Expect(testCtx.Cli.List(testCtx.Ctx, vscList)).Should(gomega.Succeed())
+ for i, vsc := range vscList.Items {
+ if vsc.Driver == storageClass.Provisioner {
+ return &vscList.Items[i]
+ }
+ }
+ return nil
+}
+
+func createVolumeSnapshotClass(testCtx *testutil.TestContext, storageClass *storagev1.StorageClass) *snapshotv1.VolumeSnapshotClass {
+ vsc := &snapshotv1.VolumeSnapshotClass{
+ ObjectMeta: metav1.ObjectMeta{
+ Name: defaultVolumeSnapshotClassName,
+ },
+ Driver: storageClass.Provisioner,
+ DeletionPolicy: snapshotv1.VolumeSnapshotContentDelete,
+ }
+ gomega.Expect(testCtx.Cli.Create(testCtx.Ctx, vsc)).Should(gomega.Succeed())
+ return vsc
+}
+
+func deleteVolumeSnapshotClass(testCtx *testutil.TestContext, storageClass *storagev1.StorageClass) {
+ vsc := getVolumeSnapshotClass(testCtx, storageClass)
+ if vsc != nil {
+ gomega.Expect(testCtx.Cli.Delete(testCtx.Ctx, vsc)).Should(gomega.Succeed())
+ }
+}
diff --git a/internal/webhook/pod_admission.go b/internal/webhook/pod_admission.go
deleted file mode 100644
index 44846baa531..00000000000
--- a/internal/webhook/pod_admission.go
+++ /dev/null
@@ -1,82 +0,0 @@
-/*
-Copyright (C) 2022-2023 ApeCloud Co., Ltd
-
-This file is part of KubeBlocks project
-
-This program is free software: you can redistribute it and/or modify
-it under the terms of the GNU Affero General Public License as published by
-the Free Software Foundation, either version 3 of the License, or
-(at your option) any later version.
-
-This program is distributed in the hope that it will be useful
-but WITHOUT ANY WARRANTY; without even the implied warranty of
-MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
-GNU Affero General Public License for more details.
-
-You should have received a copy of the GNU Affero General Public License
-along with this program. If not, see .
-*/
-
-package webhook
-
-import (
- "context"
- "net/http"
-
- corev1 "k8s.io/api/core/v1"
- "sigs.k8s.io/controller-runtime/pkg/client"
- "sigs.k8s.io/controller-runtime/pkg/runtime/inject"
- "sigs.k8s.io/controller-runtime/pkg/webhook/admission"
-)
-
-// PodCreateHandler handles Pod
-type PodCreateHandler struct {
- Client client.Client
- Decoder *admission.Decoder
-}
-
-func init() {
- HandlerMap["/mutate-v1-pod"] = &PodCreateHandler{}
-}
-
-var _ admission.Handler = &PodCreateHandler{}
-
-// Handle handles admission requests.
-func (h *PodCreateHandler) Handle(ctx context.Context, req admission.Request) admission.Response {
- pod := &corev1.Pod{}
-
- err := h.Decoder.Decode(req, pod)
- if err != nil {
- return admission.Errored(http.StatusBadRequest, err)
- }
- // mutate the fields in pod
-
- // // when pod.namespace is empty, using req.namespace
- // if pod.Namespace == "" {
- // pod.Namespace = req.Namespace
- // }
-
- // marshaledPod, err := json.Marshal(pod)
- // if err != nil {
- // return admission.Errored(http.StatusInternalServerError, err)
- // }
- // return admission.PatchResponseFromRaw(req.Object.Raw, marshaledPod)
-
- return admission.Allowed("")
-}
-
-var _ inject.Client = &PodCreateHandler{}
-
-// InjectClient injects the client into the PodCreateHandler
-func (h *PodCreateHandler) InjectClient(c client.Client) error {
- h.Client = c
- return nil
-}
-
-var _ admission.DecoderInjector = &PodCreateHandler{}
-
-// InjectDecoder injects the decoder into the PodCreateHandler
-func (h *PodCreateHandler) InjectDecoder(d *admission.Decoder) error {
- h.Decoder = d
- return nil
-}
diff --git a/lorry/binding/base.go b/lorry/binding/base.go
index 88203da0aa6..672f8f36c36 100644
--- a/lorry/binding/base.go
+++ b/lorry/binding/base.go
@@ -41,13 +41,6 @@ type Operation func(ctx context.Context, req *ProbeRequest, resp *ProbeResponse)
type OpsResult map[string]interface{}
-type GlobalInfo struct {
- Event string `json:"event,omitempty"`
- Term int `json:"term,omitempty"`
- PodName2Role map[string]string `json:"map,omitempty"`
- Message string `json:"message,omitempty"`
-}
-
// AccessMode defines SVC access mode enums.
// +enum
type AccessMode string
@@ -76,14 +69,12 @@ type BaseOperations struct {
DBAddress string
DBType string
OriRole string
- OriGlobalInfo *GlobalInfo
DBRoles map[string]AccessMode
Logger logr.Logger
Metadata map[string]string
InitIfNeed func() bool
Manager component.DBManager
GetRole func(context.Context, *ProbeRequest, *ProbeResponse) (string, error)
- GetGlobalInfo func(ctx context.Context, request *ProbeRequest, response *ProbeResponse) (GlobalInfo, error)
OperationsMap map[OperationKind]Operation
}
@@ -204,6 +195,7 @@ func (ops *BaseOperations) Invoke(ctx context.Context, req *ProbeRequest) (*Prob
if err != nil {
return nil, err
}
+ ops.Logger.Info("operation called", "operation", req.Operation, "result", opsRes)
if opsRes != nil {
res, _ := json.Marshal(opsRes)
resp.Data = res
@@ -225,8 +217,14 @@ func (ops *BaseOperations) CheckRoleOps(ctx context.Context, req *ProbeRequest,
return opsRes, nil
}
- // sql exec timeout needs to be less than httpget's timeout which by default 1s.
- ctx1, cancel := context.WithTimeout(ctx, 500*time.Millisecond)
+ timeoutSeconds := defaultRoleProbeTimeoutSeconds
+ if viper.IsSet(roleProbeTimeoutVarName) {
+ timeoutSeconds = viper.GetInt(roleProbeTimeoutVarName)
+ }
+ // lorry utilizes the pod readiness probe to trigger role probe and 'timeoutSeconds' is directly copied from the 'probe.timeoutSeconds' field of pod.
+ // here we give 80% of the total time to role probe job and leave the remaining 20% to kubelet to handle the readiness probe related tasks.
+ timeout := time.Duration(timeoutSeconds) * (800 * time.Millisecond)
+ ctx1, cancel := context.WithTimeout(ctx, timeout)
defer cancel()
role, err := ops.GetRole(ctx1, req, resp)
if err != nil {
@@ -235,7 +233,7 @@ func (ops *BaseOperations) CheckRoleOps(ctx context.Context, req *ProbeRequest,
opsRes["message"] = err.Error()
if ops.CheckRoleFailedCount%ops.FailedEventReportFrequency == 0 {
ops.Logger.Info("role checks failed continuously", "times", ops.CheckRoleFailedCount)
- SentProbeEvent(ctx, opsRes, ops.Logger)
+ SentProbeEvent(ctx, opsRes, resp, ops.Logger)
}
ops.CheckRoleFailedCount++
return opsRes, nil
@@ -252,7 +250,7 @@ func (ops *BaseOperations) CheckRoleOps(ctx context.Context, req *ProbeRequest,
opsRes["role"] = role
if ops.OriRole != role {
ops.OriRole = role
- SentProbeEvent(ctx, opsRes, ops.Logger)
+ SentProbeEvent(ctx, opsRes, resp, ops.Logger)
}
// RoleUnchangedCount is the count of consecutive role unchanged checks.
@@ -294,51 +292,6 @@ func (ops *BaseOperations) GetRoleOps(ctx context.Context, req *ProbeRequest, re
return opsRes, nil
}
-func (ops *BaseOperations) GetGlobalInfoOps(ctx context.Context, req *ProbeRequest, resp *ProbeResponse) (OpsResult, error) {
- opsRes := OpsResult{}
- opsRes["operation"] = GetGlobalInfoOperation
- if ops.GetGlobalInfo == nil {
- message := fmt.Sprintf("getGlobalInfo operation is not implemented for %v", ops.DBType)
- ops.Logger.Error(fmt.Errorf("not implemented"), message)
- opsRes["event"] = OperationNotImplemented
- opsRes["message"] = message
- resp.Metadata[StatusCode] = OperationNotFoundHTTPCode
- return opsRes, nil
- }
-
- globalInfo, err := ops.GetGlobalInfo(ctx, req, resp)
- if err != nil {
- ops.Logger.Error(err, "error executing GlobalInfo")
- opsRes["event"] = OperationFailed
- opsRes["message"] = err.Error()
- if ops.CheckRoleFailedCount%ops.FailedEventReportFrequency == 0 {
- ops.Logger.Info("getRole failed continuously", "failed times", ops.CheckRoleFailedCount)
- SentProbeEvent(ctx, opsRes, ops.Logger)
- }
- // just reuse the checkRoleFailCount temporarily
- ops.CheckRoleFailedCount++
- return opsRes, nil
- }
-
- ops.CheckRoleFailedCount = 0
-
- for _, role := range globalInfo.PodName2Role {
- if isValid, message := ops.roleValidate(role); !isValid {
- opsRes["event"] = OperationInvalid
- opsRes["message"] = message
- return opsRes, nil
- }
- }
-
- globalInfo.Transform(opsRes)
- if ops.OriGlobalInfo == nil || globalInfo.ShouldUpdate(*ops.OriGlobalInfo) {
- ops.OriGlobalInfo = &globalInfo
- SentProbeEvent(ctx, opsRes, ops.Logger)
- }
-
- return opsRes, nil
-}
-
// Component may have some internal roles that needn't be exposed to end user,
// and not configured in cluster definition, e.g. ETCD's Candidate.
// roleValidate is used to filter the internal roles and decrease the number
@@ -381,7 +334,7 @@ func (ops *BaseOperations) CheckRunningOps(ctx context.Context, req *ProbeReques
if ops.CheckRunningFailedCount%ops.FailedEventReportFrequency == 0 {
ops.Logger.Info("running checks failed continuously", "times", ops.CheckRunningFailedCount)
// resp.Metadata[StatusCode] = OperationFailedHTTPCode
- SentProbeEvent(ctx, opsRes, ops.Logger)
+ SentProbeEvent(ctx, opsRes, resp, ops.Logger)
}
ops.CheckRunningFailedCount++
return opsRes, nil
@@ -476,10 +429,11 @@ func (ops *BaseOperations) SwitchoverOps(ctx context.Context, req *ProbeRequest,
func (ops *BaseOperations) JoinMemberOps(ctx context.Context, req *ProbeRequest, resp *ProbeResponse) (OpsResult, error) {
opsRes := OpsResult{}
manager, err := component.GetDefaultManager()
- if err != nil {
- opsRes["event"] = OperationFailed
+ if manager == nil {
+ // manager for the DB is not supported, just return
+ opsRes["event"] = OperationSuccess
opsRes["message"] = err.Error()
- return opsRes, err
+ return opsRes, nil
}
dcsStore := dcs.GetStore()
@@ -515,10 +469,11 @@ func (ops *BaseOperations) JoinMemberOps(ctx context.Context, req *ProbeRequest,
func (ops *BaseOperations) LeaveMemberOps(ctx context.Context, req *ProbeRequest, resp *ProbeResponse) (OpsResult, error) {
opsRes := OpsResult{}
manager, err := component.GetDefaultManager()
- if err != nil {
- opsRes["event"] = OperationFailed
+ if manager == nil {
+ // manager for the DB is not supported, just return
+ opsRes["event"] = OperationSuccess
opsRes["message"] = err.Error()
- return opsRes, err
+ return opsRes, nil
}
dcsStore := dcs.GetStore()
@@ -549,29 +504,3 @@ func (ops *BaseOperations) LeaveMemberOps(ctx context.Context, req *ProbeRequest
opsRes["message"] = "left of the current member is complete"
return opsRes, nil
}
-
-func (g *GlobalInfo) ShouldUpdate(another GlobalInfo) bool {
- if g.Term != another.Term {
- return g.Term < another.Term
- }
- if g.Message != another.Message || g.Event != another.Event {
- return true
- }
- for k, v := range g.PodName2Role {
- if s, ok := another.PodName2Role[k]; ok {
- if s != v {
- return true
- }
- } else {
- return true
- }
- }
- return false
-}
-
-func (g *GlobalInfo) Transform(result OpsResult) {
- result["event"] = g.Event
- result["term"] = g.Term
- result["message"] = g.Message
- result["map"] = g.PodName2Role
-}
diff --git a/lorry/binding/custom/custom.go b/lorry/binding/custom/custom.go
index 5f87868e169..0de70843e22 100644
--- a/lorry/binding/custom/custom.go
+++ b/lorry/binding/custom/custom.go
@@ -27,12 +27,13 @@ import (
"net"
"net/http"
"net/url"
- "strconv"
+ "regexp"
"strings"
"time"
ctrl "sigs.k8s.io/controller-runtime"
+ "github.com/apecloud/kubeblocks/internal/common"
viper "github.com/apecloud/kubeblocks/internal/viperx"
. "github.com/apecloud/kubeblocks/lorry/binding"
"github.com/apecloud/kubeblocks/lorry/component"
@@ -46,6 +47,8 @@ type HTTPCustom struct {
BaseOperations
}
+var perNodeRegx = regexp.MustCompile("[a-zA-Z0-9]+")
+
// NewHTTPCustom returns a new HTTPCustom.
func NewHTTPCustom() *HTTPCustom {
logger := ctrl.Log.WithName("Custom")
@@ -81,9 +84,7 @@ func (h *HTTPCustom) Init(metadata component.Properties) error {
h.BaseOperations.Init(metadata)
h.BaseOperations.GetRole = h.GetRole
- h.BaseOperations.GetGlobalInfo = h.GetGlobalInfo
h.OperationsMap[CheckRoleOperation] = h.CheckRoleOps
- h.OperationsMap[GetGlobalInfoOperation] = h.GetGlobalInfoOps
return nil
}
@@ -99,73 +100,49 @@ func (h *HTTPCustom) GetRole(ctx context.Context, req *ProbeRequest, resp *Probe
)
for _, port := range *h.actionSvcPorts {
- u := fmt.Sprintf("http://127.0.0.1:%d/role?KB_CONSENSUS_SET_LAST_STDOUT=%s", port, url.QueryEscape(string(lastOutput)))
+ u := fmt.Sprintf("http://127.0.0.1:%d/role?KB_RSM_LAST_STDOUT=%s", port, url.QueryEscape(string(lastOutput)))
lastOutput, err = h.callAction(ctx, u)
if err != nil {
return "", err
}
+ h.Logger.Info("action succeed", "url", u, "output", string(lastOutput))
}
+ finalOutput := strings.TrimSpace(string(lastOutput))
- return string(lastOutput), nil
-}
-
-func (h *HTTPCustom) GetRoleOps(ctx context.Context, req *ProbeRequest, resp *ProbeResponse) (OpsResult, error) {
- role, err := h.GetRole(ctx, req, resp)
- if err != nil {
- return nil, err
- }
- opsRes := OpsResult{}
- opsRes["role"] = role
- return opsRes, nil
-}
-
-func (h *HTTPCustom) GetGlobalInfo(ctx context.Context, req *ProbeRequest, resp *ProbeResponse) (GlobalInfo, error) {
- if h.actionSvcPorts == nil {
- return GlobalInfo{}, nil
+ if perNodeRegx.MatchString(finalOutput) {
+ return finalOutput, nil
}
- var (
- lastOutput []byte
- err error
- )
-
- for _, port := range *h.actionSvcPorts {
- u := fmt.Sprintf("http://127.0.0.1:%d/role?KB_CONSENSUS_SET_LAST_STDOUT=%s", port, url.QueryEscape(string(lastOutput)))
- lastOutput, err = h.callAction(ctx, u)
- if err != nil {
- return GlobalInfo{}, err
- }
- }
-
- // csv format: term,podname,role
- parseCSV := func(input []byte) (GlobalInfo, error) {
- res := GlobalInfo{PodName2Role: map[string]string{}}
- str := string(input)
- lines := strings.Split(str, "\n")
+ // csv format: term,podName,role
+ parseCSV := func(input string) (string, error) {
+ res := common.GlobalRoleSnapshot{}
+ lines := strings.Split(input, "\n")
for _, line := range lines {
- fields := strings.Split(line, ",")
+ fields := strings.Split(strings.TrimSpace(line), ",")
if len(fields) != 3 {
- return res, err
+ return "", err
}
- res.Term, err = strconv.Atoi(fields[0])
- if err != nil {
- return res, err
+ res.Version = strings.TrimSpace(fields[0])
+ pair := common.PodRoleNamePair{
+ PodName: strings.TrimSpace(fields[1]),
+ RoleName: strings.ToLower(strings.TrimSpace(fields[2])),
}
- k := fields[1]
- v := fields[2]
- res.PodName2Role[k] = v
+ res.PodRoleNamePairs = append(res.PodRoleNamePairs, pair)
}
- return res, nil
+ resByte, err := json.Marshal(res)
+ return string(resByte), err
}
+ return parseCSV(finalOutput)
+}
- res, err := parseCSV(lastOutput)
+func (h *HTTPCustom) GetRoleOps(ctx context.Context, req *ProbeRequest, resp *ProbeResponse) (OpsResult, error) {
+ role, err := h.GetRole(ctx, req, resp)
if err != nil {
- return GlobalInfo{}, err
+ return nil, err
}
- res.Event = OperationSuccess
- h.Logger.Info("GetGlobalInfo get result", "result", res)
-
- return res, nil
+ opsRes := OpsResult{}
+ opsRes["role"] = role
+ return opsRes, nil
}
// callAction performs an HTTP request to local HTTP endpoint specified by actionSvcPort
diff --git a/lorry/binding/custom/custom_test.go b/lorry/binding/custom/custom_test.go
index 06441ae24ca..c9f88a496f7 100644
--- a/lorry/binding/custom/custom_test.go
+++ b/lorry/binding/custom/custom_test.go
@@ -21,6 +21,7 @@ package custom
import (
"context"
+ "encoding/json"
"fmt"
"net/http"
"net/http/httptest"
@@ -31,6 +32,7 @@ import (
"github.com/stretchr/testify/assert"
"github.com/stretchr/testify/require"
+ "github.com/apecloud/kubeblocks/internal/common"
viper "github.com/apecloud/kubeblocks/internal/viperx"
"github.com/apecloud/kubeblocks/lorry/binding"
"github.com/apecloud/kubeblocks/lorry/component"
@@ -75,7 +77,7 @@ func TestInit(t *testing.T) {
}
}
-func TestGlobalInfo(t *testing.T) {
+func TestGlobalRoleSnapshot(t *testing.T) {
var lines []string
for i := 0; i < 3; i++ {
podName := "pod-" + strconv.Itoa(i)
@@ -100,7 +102,7 @@ func TestGlobalInfo(t *testing.T) {
}{
"get": {
input: join,
- operation: "getGlobalInfo",
+ operation: "getRole",
metadata: nil,
path: "/",
err: "",
@@ -110,13 +112,14 @@ func TestGlobalInfo(t *testing.T) {
for name, tc := range tests {
t.Run(name, func(t *testing.T) {
response := binding.ProbeResponse{}
- info, err := hs.GetGlobalInfo(context.TODO(), &binding.ProbeRequest{
+ info, err := hs.GetRole(context.TODO(), &binding.ProbeRequest{
Operation: OperationKind(tc.operation),
}, &response)
require.NoError(t, err)
- assert.Equal(t, OperationSuccess, info.Event)
- assert.Equal(t, 3, len(info.PodName2Role))
- assert.Equal(t, 1, info.Term)
+ snapshot := &common.GlobalRoleSnapshot{}
+ assert.NoError(t, json.Unmarshal([]byte(info), snapshot))
+ assert.Equal(t, 3, len(snapshot.PodRoleNamePairs))
+ assert.Equal(t, "1", snapshot.Version)
})
}
diff --git a/lorry/binding/etcd/etcd_test.go b/lorry/binding/etcd/etcd_test.go
index 0ec8e460638..1c6cbf35da0 100644
--- a/lorry/binding/etcd/etcd_test.go
+++ b/lorry/binding/etcd/etcd_test.go
@@ -112,8 +112,8 @@ func (e *EmbeddedETCD) Start(peerAddress string) error {
cfg.Dir = dir
lpurl, _ := url.Parse("http://localhost:0")
lcurl, _ := url.Parse(peerAddress)
- cfg.LPUrls = []url.URL{*lpurl}
- cfg.LCUrls = []url.URL{*lcurl}
+ cfg.ListenPeerUrls = []url.URL{*lpurl}
+ cfg.ListenClientUrls = []url.URL{*lcurl}
e.ETCD, err = embed.StartEtcd(cfg)
if err != nil {
return err
diff --git a/lorry/binding/mysql/mysql.go b/lorry/binding/mysql/mysql.go
index 51ee885c9d6..80c161e9c51 100644
--- a/lorry/binding/mysql/mysql.go
+++ b/lorry/binding/mysql/mysql.go
@@ -106,7 +106,6 @@ func (mysqlOps *MysqlOperations) Init(metadata component.Properties) error {
mysqlOps.DBType = "mysql"
// mysqlOps.InitIfNeed = mysqlOps.initIfNeed
mysqlOps.BaseOperations.GetRole = mysqlOps.GetRole
- mysqlOps.BaseOperations.GetGlobalInfo = mysqlOps.GetGlobalInfo
mysqlOps.DBPort = config.GetDBPort()
mysqlOps.RegisterOperationOnDBReady(GetRoleOperation, mysqlOps.GetRoleOps, manager)
@@ -115,7 +114,6 @@ func (mysqlOps *MysqlOperations) Init(metadata component.Properties) error {
mysqlOps.RegisterOperationOnDBReady(CheckStatusOperation, mysqlOps.CheckStatusOps, manager)
mysqlOps.RegisterOperationOnDBReady(ExecOperation, mysqlOps.ExecOps, manager)
mysqlOps.RegisterOperationOnDBReady(QueryOperation, mysqlOps.QueryOps, manager)
- mysqlOps.RegisterOperationOnDBReady(GetGlobalInfoOperation, mysqlOps.GetGlobalInfoOps, manager)
// following are ops for account management
mysqlOps.RegisterOperationOnDBReady(ListUsersOp, mysqlOps.listUsersOps, manager)
diff --git a/lorry/binding/types.go b/lorry/binding/types.go
index 348e912bd3f..95fd547f222 100644
--- a/lorry/binding/types.go
+++ b/lorry/binding/types.go
@@ -49,6 +49,10 @@ const (
roleEventReportFrequency = int(1 / roleEventRecordQPS)
defaultFailedEventReportFrequency = 1800
defaultRoleDetectionThreshold = 300
+ defaultRoleProbeTimeoutSeconds = 2
+
+ rsmRoleUpdateMechanismVarName = "KB_RSM_ROLE_UPDATE_MECHANISM"
+ roleProbeTimeoutVarName = "KB_RSM_ROLE_PROBE_TIMEOUT"
)
const (
diff --git a/lorry/binding/utils.go b/lorry/binding/utils.go
index 6e7dfcd01b1..ffad3301a8b 100644
--- a/lorry/binding/utils.go
+++ b/lorry/binding/utils.go
@@ -37,7 +37,9 @@ import (
"k8s.io/client-go/kubernetes/scheme"
ctlruntime "sigs.k8s.io/controller-runtime"
+ workloads "github.com/apecloud/kubeblocks/apis/workloads/v1alpha1"
"github.com/apecloud/kubeblocks/internal/constant"
+ viper "github.com/apecloud/kubeblocks/internal/viperx"
"github.com/apecloud/kubeblocks/lorry/component"
. "github.com/apecloud/kubeblocks/lorry/util"
)
@@ -220,15 +222,26 @@ func String2RoleType(roleName string) RoleType {
return CustomizedRole
}
-func SentProbeEvent(ctx context.Context, opsResult OpsResult, log logr.Logger) {
+func SentProbeEvent(ctx context.Context, opsResult OpsResult, resp *ProbeResponse, log logr.Logger) {
log.Info(fmt.Sprintf("send event: %v", opsResult))
- event, err := createProbeEvent(opsResult)
- if err != nil {
- log.Error(err, "generate event failed")
- return
+ roleUpdateMechanism := workloads.NoneUpdate
+ if viper.IsSet(rsmRoleUpdateMechanismVarName) {
+ roleUpdateMechanism = workloads.RoleUpdateMechanism(viper.GetString(rsmRoleUpdateMechanismVarName))
}
+ switch roleUpdateMechanism {
+ case workloads.ReadinessProbeEventUpdate:
+ resp.Metadata[StatusCode] = OperationFailedHTTPCode
+ case workloads.DirectAPIServerEventUpdate:
+ event, err := createProbeEvent(opsResult)
+ if err != nil {
+ log.Error(err, "generate event failed")
+ return
+ }
- _ = sendEvent(ctx, log, event)
+ _ = sendEvent(ctx, log, event)
+ default:
+ log.Info(fmt.Sprintf("no event sent, RoleUpdateMechanism: %s", roleUpdateMechanism))
+ }
}
func createProbeEvent(opsResult OpsResult) (*corev1.Event, error) {
diff --git a/lorry/client/client.go b/lorry/client/client.go
index 5712fef0bc9..ac4867398a1 100644
--- a/lorry/client/client.go
+++ b/lorry/client/client.go
@@ -39,7 +39,7 @@ import (
)
const (
- urlTemplate = "http://localhost:%d/v1.0/bindings/%s"
+ urlTemplate = "http://%s:%d/v1.0/bindings/%s"
)
type Client interface {
@@ -100,9 +100,10 @@ func NewClientWithPod(pod *corev1.Pod, characterType string) (*OperationClient,
return nil, fmt.Errorf("pod %v has no ip", pod.Name)
}
- port, err := intctrlutil.GetProbeHTTPPort(pod)
+ port, err := intctrlutil.GuessLorryHTTPPort(pod)
if err != nil {
- return nil, err
+ // not lorry in the pod, just return nil without error
+ return nil, nil
}
// don't use default http-client
@@ -122,7 +123,7 @@ func NewClientWithPod(pod *corev1.Pod, characterType string) (*OperationClient,
Client: client,
Port: port,
CharacterType: characterType,
- URL: fmt.Sprintf(urlTemplate, port, characterType),
+ URL: fmt.Sprintf(urlTemplate, ip, port, characterType),
CacheTTL: 60 * time.Second,
RequestTimeout: 30 * time.Second,
ReconcileTimeout: 500 * time.Millisecond,
diff --git a/lorry/component/dbmanager.go b/lorry/component/dbmanager.go
index 2ed4db855b9..9406e351388 100644
--- a/lorry/component/dbmanager.go
+++ b/lorry/component/dbmanager.go
@@ -222,6 +222,14 @@ func (mgr *DBManagerBase) ShutDownWithWait() {
mgr.Logger.Info("Override me if need")
}
+func (*DBManagerBase) JoinCurrentMemberToCluster(context.Context, *dcs.Cluster) error {
+ return nil
+}
+
+func (*DBManagerBase) LeaveMemberFromCluster(context.Context, *dcs.Cluster, string) error {
+ return nil
+}
+
func RegisterManager(characterType, workloadType string, manager DBManager) {
key := strings.ToLower(characterType + "_" + workloadType)
managers[key] = manager
@@ -306,11 +314,11 @@ func (*FakeManager) IsFirstMember() bool {
}
func (*FakeManager) JoinCurrentMemberToCluster(context.Context, *dcs.Cluster) error {
- return fmt.Errorf("NotSupported")
+ return nil
}
func (*FakeManager) LeaveMemberFromCluster(context.Context, *dcs.Cluster, string) error {
- return fmt.Errorf("NotSuppported")
+ return nil
}
func (*FakeManager) Promote(context.Context, *dcs.Cluster) error {
diff --git a/lorry/engine/engine_test.go b/lorry/engine/engine_test.go
index 599f54b61e1..d943a59af6e 100644
--- a/lorry/engine/engine_test.go
+++ b/lorry/engine/engine_test.go
@@ -53,14 +53,14 @@ var _ = Describe("Engine", func() {
})
It("new execute command ", func() {
- for _, typeName := range []string{stateMysql, statePostgreSQL} {
+ for _, typeName := range []string{stateMysql, statePostgreSQL, stateRedis} {
engine, _ := New(typeName)
Expect(engine).ShouldNot(BeNil())
_, _, err := engine.ExecuteCommand([]string{"some", "cmd"})
Expect(err).Should(Succeed())
}
- for _, typeName := range []string{stateRedis, stateMongoDB, stateNebula} {
+ for _, typeName := range []string{stateMongoDB, stateNebula} {
engine, _ := New(typeName)
Expect(engine).ShouldNot(BeNil())
diff --git a/lorry/engine/redis.go b/lorry/engine/redis.go
index 409bed81388..8228a2639c6 100644
--- a/lorry/engine/redis.go
+++ b/lorry/engine/redis.go
@@ -69,6 +69,16 @@ func (r redis) ConnectExample(info *ConnectionInfo, client string) string {
return buildExample(info, client, r.examples)
}
-func (r redis) ExecuteCommand([]string) ([]string, []corev1.EnvVar, error) {
- return nil, nil, fmt.Errorf("%s not implemented", r.info.Client)
+func (r redis) ExecuteCommand(scripts []string) ([]string, []corev1.EnvVar, error) {
+ cmd := []string{}
+ args := []string{}
+ cmd = append(cmd, "/bin/sh", "-c")
+ for _, script := range scripts {
+ args = append(args, fmt.Sprintf("%s -h %s -p 6379 --user %s --pass %s %s", r.info.Client,
+ fmt.Sprintf("$%s", envVarMap[host]),
+ fmt.Sprintf("$%s", envVarMap[user]),
+ fmt.Sprintf("$%s", envVarMap[password]), script))
+ }
+ cmd = append(cmd, strings.Join(args, " && "))
+ return cmd, nil, nil
}
diff --git a/lorry/highavailability/ha.go b/lorry/highavailability/ha.go
index 6f3a2761412..b8b05fbb9d7 100644
--- a/lorry/highavailability/ha.go
+++ b/lorry/highavailability/ha.go
@@ -413,6 +413,12 @@ func (ha *Ha) IsPodReady() (bool, error) {
pinger.Interval = 500 * time.Millisecond
err = pinger.Run()
if err != nil {
+ // For container runtimes like Containerd, unprivileged users can't send icmp echo packets.
+ // As a temporary workaround, special handling is being implemented to bypass this limitation.
+ if strings.Contains(err.Error(), "socket: permission denied") {
+ ha.logger.Info("ping failed, socket: permission denied, but temporarily return true")
+ return true, nil
+ }
ha.logger.Error(err, fmt.Sprintf("ping domain:%s failed", domain))
return false, err
}
diff --git a/lorry/middleware/http/probe/checks_middleware.go b/lorry/middleware/http/probe/checks_middleware.go
index e5fce9b1373..614aa63031f 100644
--- a/lorry/middleware/http/probe/checks_middleware.go
+++ b/lorry/middleware/http/probe/checks_middleware.go
@@ -28,8 +28,7 @@ import (
"strings"
"github.com/go-logr/logr"
- "github.com/go-logr/zapr"
- "go.uber.org/zap"
+ ctrl "sigs.k8s.io/controller-runtime"
)
const (
@@ -48,14 +47,10 @@ type RequestMeta struct {
Metadata map[string]string `json:"metadata"`
}
-var Logger logr.Logger
+var logger logr.Logger
func init() {
- development, err := zap.NewProduction()
- if err != nil {
- panic(err)
- }
- Logger = zapr.NewLogger(development)
+ logger = ctrl.Log.WithName("middleware")
}
func GetRequestBody(operation string, args map[string][]string) []byte {
@@ -69,7 +64,7 @@ func GetRequestBody(operation string, args map[string][]string) []byte {
} else {
marshal, err := json.Marshal(value)
if err != nil {
- Logger.Error(err, "getRequestBody marshal json error")
+ logger.Error(err, "getRequestBody marshal json error")
return
}
metadata[key] = string(marshal)
@@ -102,21 +97,21 @@ func SetMiddleware(next http.HandlerFunc) http.HandlerFunc {
body := GetRequestBody(operation, uri.Query())
request.Body = io.NopCloser(bytes.NewReader(body))
} else {
- Logger.Info("unknown probe operation", "operation", operation)
+ logger.Info("unknown probe operation", "operation", operation)
}
}
- Logger.Info("receive request", "request", request.RequestURI)
+ logger.Info("receive request", "request", request.RequestURI)
next(writer, request)
code := writer.Header().Get(statusCodeHeader)
statusCode, err := strconv.Atoi(code)
if err == nil {
// header has a statusCodeHeader
writer.WriteHeader(statusCode)
- Logger.Info("response abnormal")
+ logger.Info("write response with header", "statusCode", statusCode)
} else {
// header has no statusCodeHeader
- Logger.Info("response has no statusCodeHeader")
+ logger.Info("response has no statusCodeHeader")
}
}
}
diff --git a/lorry/middleware/http/probe/router.go b/lorry/middleware/http/probe/router.go
index d49a315e809..66bb03a44e8 100644
--- a/lorry/middleware/http/probe/router.go
+++ b/lorry/middleware/http/probe/router.go
@@ -101,7 +101,7 @@ func GetRouter() func(writer http.ResponseWriter, request *http.Request) {
// get the character type
character := GetCharacter(request.URL.Path)
if character == "" {
- Logger.Error(nil, "character type missing in path")
+ logger.Error(nil, "character type missing in path")
return
}
@@ -109,14 +109,14 @@ func GetRouter() func(writer http.ResponseWriter, request *http.Request) {
defer body.Close()
buf, err := io.ReadAll(request.Body)
if err != nil {
- Logger.Error(err, "request body read failed")
+ logger.Error(err, "request body read failed")
return
}
meta := &RequestMeta{Metadata: map[string]string{}}
err = json.Unmarshal(buf, meta)
if err != nil {
- Logger.Error(err, "request body unmarshal failed")
+ logger.Error(err, "request body unmarshal failed")
return
}
probeRequest := &ProbeRequest{Metadata: meta.Metadata}
@@ -124,14 +124,15 @@ func GetRouter() func(writer http.ResponseWriter, request *http.Request) {
// route the request to engine
probeResp, err := route(character, request.Context(), probeRequest)
+ logger.Info("request routed", "request", probeRequest, "response", probeResp)
if err != nil {
- Logger.Error(err, "exec ops failed")
+ logger.Error(err, "exec ops failed")
msg := fmt.Sprintf("exec ops failed: %v", err)
writer.Header().Add(statusCodeHeader, OperationFailedHTTPCode)
_, err := writer.Write([]byte(msg))
if err != nil {
- Logger.Error(err, "ResponseWriter writes error when router")
+ logger.Error(err, "ResponseWriter writes error when router")
}
} else {
code, ok := probeResp.Metadata[StatusCode]
@@ -142,7 +143,7 @@ func GetRouter() func(writer http.ResponseWriter, request *http.Request) {
writer.Header().Add(RespEndTimeKey, probeResp.Metadata[RespEndTimeKey])
_, err := writer.Write(probeResp.Data)
if err != nil {
- Logger.Error(err, "ResponseWriter writes error when router")
+ logger.Error(err, "ResponseWriter writes error when router")
}
}
}
@@ -163,7 +164,7 @@ func route(character string, ctx context.Context, request *ProbeRequest) (*Probe
ops, ok := builtinMap[character]
// if there is no builtin type, use the custom
if !ok {
- Logger.Info("No correspond builtin type, use the custom...")
+ logger.Info("No correspond builtin type, use the custom...")
return customOp.Invoke(ctx, request)
}
return ops.Invoke(ctx, request)
diff --git a/lorry/util/types.go b/lorry/util/types.go
index 3a82c5c477f..ee6151cec2b 100644
--- a/lorry/util/types.go
+++ b/lorry/util/types.go
@@ -48,10 +48,9 @@ const (
QueryOperation OperationKind = "query"
CloseOperation OperationKind = "close"
- LockOperation OperationKind = "lockInstance"
- UnlockOperation OperationKind = "unlockInstance"
- VolumeProtection OperationKind = "volumeProtection"
- GetGlobalInfoOperation OperationKind = "getGlobalInfo"
+ LockOperation OperationKind = "lockInstance"
+ UnlockOperation OperationKind = "unlockInstance"
+ VolumeProtection OperationKind = "volumeProtection"
// actions for cluster accounts management
ListUsersOp OperationKind = "listUsers"
diff --git a/pkg/client/clientset/versioned/doc.go b/pkg/client/clientset/versioned/doc.go
deleted file mode 100644
index d4e0d371e5b..00000000000
--- a/pkg/client/clientset/versioned/doc.go
+++ /dev/null
@@ -1,20 +0,0 @@
-/*
-Copyright (C) 2022-2023 ApeCloud Co., Ltd
-
-Licensed under the Apache License, Version 2.0 (the "License");
-you may not use this file except in compliance with the License.
-You may obtain a copy of the License at
-
- http://www.apache.org/licenses/LICENSE-2.0
-
-Unless required by applicable law or agreed to in writing, software
-distributed under the License is distributed on an "AS IS" BASIS,
-WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
-See the License for the specific language governing permissions and
-limitations under the License.
-*/
-
-// Code generated by client-gen. DO NOT EDIT.
-
-// This package has the automatically generated clientset.
-package versioned
diff --git a/pkg/client/clientset/versioned/typed/apps/v1alpha1/fake/fake_backuppolicytemplate.go b/pkg/client/clientset/versioned/typed/apps/v1alpha1/fake/fake_backuppolicytemplate.go
index 007195219cc..bc12fb95324 100644
--- a/pkg/client/clientset/versioned/typed/apps/v1alpha1/fake/fake_backuppolicytemplate.go
+++ b/pkg/client/clientset/versioned/typed/apps/v1alpha1/fake/fake_backuppolicytemplate.go
@@ -24,7 +24,6 @@ import (
v1alpha1 "github.com/apecloud/kubeblocks/apis/apps/v1alpha1"
v1 "k8s.io/apimachinery/pkg/apis/meta/v1"
labels "k8s.io/apimachinery/pkg/labels"
- schema "k8s.io/apimachinery/pkg/runtime/schema"
types "k8s.io/apimachinery/pkg/types"
watch "k8s.io/apimachinery/pkg/watch"
testing "k8s.io/client-go/testing"
@@ -35,9 +34,9 @@ type FakeBackupPolicyTemplates struct {
Fake *FakeAppsV1alpha1
}
-var backuppolicytemplatesResource = schema.GroupVersionResource{Group: "apps.kubeblocks.io", Version: "v1alpha1", Resource: "backuppolicytemplates"}
+var backuppolicytemplatesResource = v1alpha1.SchemeGroupVersion.WithResource("backuppolicytemplates")
-var backuppolicytemplatesKind = schema.GroupVersionKind{Group: "apps.kubeblocks.io", Version: "v1alpha1", Kind: "BackupPolicyTemplate"}
+var backuppolicytemplatesKind = v1alpha1.SchemeGroupVersion.WithKind("BackupPolicyTemplate")
// Get takes name of the backupPolicyTemplate, and returns the corresponding backupPolicyTemplate object, and an error if there is any.
func (c *FakeBackupPolicyTemplates) Get(ctx context.Context, name string, options v1.GetOptions) (result *v1alpha1.BackupPolicyTemplate, err error) {
diff --git a/pkg/client/clientset/versioned/typed/apps/v1alpha1/fake/fake_cluster.go b/pkg/client/clientset/versioned/typed/apps/v1alpha1/fake/fake_cluster.go
index 468a18b4a76..737db9f313e 100644
--- a/pkg/client/clientset/versioned/typed/apps/v1alpha1/fake/fake_cluster.go
+++ b/pkg/client/clientset/versioned/typed/apps/v1alpha1/fake/fake_cluster.go
@@ -24,7 +24,6 @@ import (
v1alpha1 "github.com/apecloud/kubeblocks/apis/apps/v1alpha1"
v1 "k8s.io/apimachinery/pkg/apis/meta/v1"
labels "k8s.io/apimachinery/pkg/labels"
- schema "k8s.io/apimachinery/pkg/runtime/schema"
types "k8s.io/apimachinery/pkg/types"
watch "k8s.io/apimachinery/pkg/watch"
testing "k8s.io/client-go/testing"
@@ -36,9 +35,9 @@ type FakeClusters struct {
ns string
}
-var clustersResource = schema.GroupVersionResource{Group: "apps.kubeblocks.io", Version: "v1alpha1", Resource: "clusters"}
+var clustersResource = v1alpha1.SchemeGroupVersion.WithResource("clusters")
-var clustersKind = schema.GroupVersionKind{Group: "apps.kubeblocks.io", Version: "v1alpha1", Kind: "Cluster"}
+var clustersKind = v1alpha1.SchemeGroupVersion.WithKind("Cluster")
// Get takes name of the cluster, and returns the corresponding cluster object, and an error if there is any.
func (c *FakeClusters) Get(ctx context.Context, name string, options v1.GetOptions) (result *v1alpha1.Cluster, err error) {
diff --git a/pkg/client/clientset/versioned/typed/apps/v1alpha1/fake/fake_clusterdefinition.go b/pkg/client/clientset/versioned/typed/apps/v1alpha1/fake/fake_clusterdefinition.go
index 0eb0185053d..f00872aaaa2 100644
--- a/pkg/client/clientset/versioned/typed/apps/v1alpha1/fake/fake_clusterdefinition.go
+++ b/pkg/client/clientset/versioned/typed/apps/v1alpha1/fake/fake_clusterdefinition.go
@@ -24,7 +24,6 @@ import (
v1alpha1 "github.com/apecloud/kubeblocks/apis/apps/v1alpha1"
v1 "k8s.io/apimachinery/pkg/apis/meta/v1"
labels "k8s.io/apimachinery/pkg/labels"
- schema "k8s.io/apimachinery/pkg/runtime/schema"
types "k8s.io/apimachinery/pkg/types"
watch "k8s.io/apimachinery/pkg/watch"
testing "k8s.io/client-go/testing"
@@ -35,9 +34,9 @@ type FakeClusterDefinitions struct {
Fake *FakeAppsV1alpha1
}
-var clusterdefinitionsResource = schema.GroupVersionResource{Group: "apps.kubeblocks.io", Version: "v1alpha1", Resource: "clusterdefinitions"}
+var clusterdefinitionsResource = v1alpha1.SchemeGroupVersion.WithResource("clusterdefinitions")
-var clusterdefinitionsKind = schema.GroupVersionKind{Group: "apps.kubeblocks.io", Version: "v1alpha1", Kind: "ClusterDefinition"}
+var clusterdefinitionsKind = v1alpha1.SchemeGroupVersion.WithKind("ClusterDefinition")
// Get takes name of the clusterDefinition, and returns the corresponding clusterDefinition object, and an error if there is any.
func (c *FakeClusterDefinitions) Get(ctx context.Context, name string, options v1.GetOptions) (result *v1alpha1.ClusterDefinition, err error) {
diff --git a/pkg/client/clientset/versioned/typed/apps/v1alpha1/fake/fake_clusterversion.go b/pkg/client/clientset/versioned/typed/apps/v1alpha1/fake/fake_clusterversion.go
index 83bcda7413a..bb34c76cf3d 100644
--- a/pkg/client/clientset/versioned/typed/apps/v1alpha1/fake/fake_clusterversion.go
+++ b/pkg/client/clientset/versioned/typed/apps/v1alpha1/fake/fake_clusterversion.go
@@ -24,7 +24,6 @@ import (
v1alpha1 "github.com/apecloud/kubeblocks/apis/apps/v1alpha1"
v1 "k8s.io/apimachinery/pkg/apis/meta/v1"
labels "k8s.io/apimachinery/pkg/labels"
- schema "k8s.io/apimachinery/pkg/runtime/schema"
types "k8s.io/apimachinery/pkg/types"
watch "k8s.io/apimachinery/pkg/watch"
testing "k8s.io/client-go/testing"
@@ -35,9 +34,9 @@ type FakeClusterVersions struct {
Fake *FakeAppsV1alpha1
}
-var clusterversionsResource = schema.GroupVersionResource{Group: "apps.kubeblocks.io", Version: "v1alpha1", Resource: "clusterversions"}
+var clusterversionsResource = v1alpha1.SchemeGroupVersion.WithResource("clusterversions")
-var clusterversionsKind = schema.GroupVersionKind{Group: "apps.kubeblocks.io", Version: "v1alpha1", Kind: "ClusterVersion"}
+var clusterversionsKind = v1alpha1.SchemeGroupVersion.WithKind("ClusterVersion")
// Get takes name of the clusterVersion, and returns the corresponding clusterVersion object, and an error if there is any.
func (c *FakeClusterVersions) Get(ctx context.Context, name string, options v1.GetOptions) (result *v1alpha1.ClusterVersion, err error) {
diff --git a/pkg/client/clientset/versioned/typed/apps/v1alpha1/fake/fake_componentclassdefinition.go b/pkg/client/clientset/versioned/typed/apps/v1alpha1/fake/fake_componentclassdefinition.go
index f0a434593af..be08339e185 100644
--- a/pkg/client/clientset/versioned/typed/apps/v1alpha1/fake/fake_componentclassdefinition.go
+++ b/pkg/client/clientset/versioned/typed/apps/v1alpha1/fake/fake_componentclassdefinition.go
@@ -24,7 +24,6 @@ import (
v1alpha1 "github.com/apecloud/kubeblocks/apis/apps/v1alpha1"
v1 "k8s.io/apimachinery/pkg/apis/meta/v1"
labels "k8s.io/apimachinery/pkg/labels"
- schema "k8s.io/apimachinery/pkg/runtime/schema"
types "k8s.io/apimachinery/pkg/types"
watch "k8s.io/apimachinery/pkg/watch"
testing "k8s.io/client-go/testing"
@@ -35,9 +34,9 @@ type FakeComponentClassDefinitions struct {
Fake *FakeAppsV1alpha1
}
-var componentclassdefinitionsResource = schema.GroupVersionResource{Group: "apps.kubeblocks.io", Version: "v1alpha1", Resource: "componentclassdefinitions"}
+var componentclassdefinitionsResource = v1alpha1.SchemeGroupVersion.WithResource("componentclassdefinitions")
-var componentclassdefinitionsKind = schema.GroupVersionKind{Group: "apps.kubeblocks.io", Version: "v1alpha1", Kind: "ComponentClassDefinition"}
+var componentclassdefinitionsKind = v1alpha1.SchemeGroupVersion.WithKind("ComponentClassDefinition")
// Get takes name of the componentClassDefinition, and returns the corresponding componentClassDefinition object, and an error if there is any.
func (c *FakeComponentClassDefinitions) Get(ctx context.Context, name string, options v1.GetOptions) (result *v1alpha1.ComponentClassDefinition, err error) {
diff --git a/pkg/client/clientset/versioned/typed/apps/v1alpha1/fake/fake_componentresourceconstraint.go b/pkg/client/clientset/versioned/typed/apps/v1alpha1/fake/fake_componentresourceconstraint.go
index f90bd2bc090..3e4c24db277 100644
--- a/pkg/client/clientset/versioned/typed/apps/v1alpha1/fake/fake_componentresourceconstraint.go
+++ b/pkg/client/clientset/versioned/typed/apps/v1alpha1/fake/fake_componentresourceconstraint.go
@@ -24,7 +24,6 @@ import (
v1alpha1 "github.com/apecloud/kubeblocks/apis/apps/v1alpha1"
v1 "k8s.io/apimachinery/pkg/apis/meta/v1"
labels "k8s.io/apimachinery/pkg/labels"
- schema "k8s.io/apimachinery/pkg/runtime/schema"
types "k8s.io/apimachinery/pkg/types"
watch "k8s.io/apimachinery/pkg/watch"
testing "k8s.io/client-go/testing"
@@ -35,9 +34,9 @@ type FakeComponentResourceConstraints struct {
Fake *FakeAppsV1alpha1
}
-var componentresourceconstraintsResource = schema.GroupVersionResource{Group: "apps.kubeblocks.io", Version: "v1alpha1", Resource: "componentresourceconstraints"}
+var componentresourceconstraintsResource = v1alpha1.SchemeGroupVersion.WithResource("componentresourceconstraints")
-var componentresourceconstraintsKind = schema.GroupVersionKind{Group: "apps.kubeblocks.io", Version: "v1alpha1", Kind: "ComponentResourceConstraint"}
+var componentresourceconstraintsKind = v1alpha1.SchemeGroupVersion.WithKind("ComponentResourceConstraint")
// Get takes name of the componentResourceConstraint, and returns the corresponding componentResourceConstraint object, and an error if there is any.
func (c *FakeComponentResourceConstraints) Get(ctx context.Context, name string, options v1.GetOptions) (result *v1alpha1.ComponentResourceConstraint, err error) {
diff --git a/pkg/client/clientset/versioned/typed/apps/v1alpha1/fake/fake_configconstraint.go b/pkg/client/clientset/versioned/typed/apps/v1alpha1/fake/fake_configconstraint.go
index c6c55dda196..5ae91311f2c 100644
--- a/pkg/client/clientset/versioned/typed/apps/v1alpha1/fake/fake_configconstraint.go
+++ b/pkg/client/clientset/versioned/typed/apps/v1alpha1/fake/fake_configconstraint.go
@@ -24,7 +24,6 @@ import (
v1alpha1 "github.com/apecloud/kubeblocks/apis/apps/v1alpha1"
v1 "k8s.io/apimachinery/pkg/apis/meta/v1"
labels "k8s.io/apimachinery/pkg/labels"
- schema "k8s.io/apimachinery/pkg/runtime/schema"
types "k8s.io/apimachinery/pkg/types"
watch "k8s.io/apimachinery/pkg/watch"
testing "k8s.io/client-go/testing"
@@ -35,9 +34,9 @@ type FakeConfigConstraints struct {
Fake *FakeAppsV1alpha1
}
-var configconstraintsResource = schema.GroupVersionResource{Group: "apps.kubeblocks.io", Version: "v1alpha1", Resource: "configconstraints"}
+var configconstraintsResource = v1alpha1.SchemeGroupVersion.WithResource("configconstraints")
-var configconstraintsKind = schema.GroupVersionKind{Group: "apps.kubeblocks.io", Version: "v1alpha1", Kind: "ConfigConstraint"}
+var configconstraintsKind = v1alpha1.SchemeGroupVersion.WithKind("ConfigConstraint")
// Get takes name of the configConstraint, and returns the corresponding configConstraint object, and an error if there is any.
func (c *FakeConfigConstraints) Get(ctx context.Context, name string, options v1.GetOptions) (result *v1alpha1.ConfigConstraint, err error) {
diff --git a/pkg/client/clientset/versioned/typed/apps/v1alpha1/fake/fake_opsrequest.go b/pkg/client/clientset/versioned/typed/apps/v1alpha1/fake/fake_opsrequest.go
index c9b181c1e96..aeb5451ec86 100644
--- a/pkg/client/clientset/versioned/typed/apps/v1alpha1/fake/fake_opsrequest.go
+++ b/pkg/client/clientset/versioned/typed/apps/v1alpha1/fake/fake_opsrequest.go
@@ -24,7 +24,6 @@ import (
v1alpha1 "github.com/apecloud/kubeblocks/apis/apps/v1alpha1"
v1 "k8s.io/apimachinery/pkg/apis/meta/v1"
labels "k8s.io/apimachinery/pkg/labels"
- schema "k8s.io/apimachinery/pkg/runtime/schema"
types "k8s.io/apimachinery/pkg/types"
watch "k8s.io/apimachinery/pkg/watch"
testing "k8s.io/client-go/testing"
@@ -36,9 +35,9 @@ type FakeOpsRequests struct {
ns string
}
-var opsrequestsResource = schema.GroupVersionResource{Group: "apps.kubeblocks.io", Version: "v1alpha1", Resource: "opsrequests"}
+var opsrequestsResource = v1alpha1.SchemeGroupVersion.WithResource("opsrequests")
-var opsrequestsKind = schema.GroupVersionKind{Group: "apps.kubeblocks.io", Version: "v1alpha1", Kind: "OpsRequest"}
+var opsrequestsKind = v1alpha1.SchemeGroupVersion.WithKind("OpsRequest")
// Get takes name of the opsRequest, and returns the corresponding opsRequest object, and an error if there is any.
func (c *FakeOpsRequests) Get(ctx context.Context, name string, options v1.GetOptions) (result *v1alpha1.OpsRequest, err error) {
diff --git a/pkg/client/clientset/versioned/typed/apps/v1alpha1/fake/fake_servicedescriptor.go b/pkg/client/clientset/versioned/typed/apps/v1alpha1/fake/fake_servicedescriptor.go
index 1e0caf12bf8..b9bddc67d41 100644
--- a/pkg/client/clientset/versioned/typed/apps/v1alpha1/fake/fake_servicedescriptor.go
+++ b/pkg/client/clientset/versioned/typed/apps/v1alpha1/fake/fake_servicedescriptor.go
@@ -24,7 +24,6 @@ import (
v1alpha1 "github.com/apecloud/kubeblocks/apis/apps/v1alpha1"
v1 "k8s.io/apimachinery/pkg/apis/meta/v1"
labels "k8s.io/apimachinery/pkg/labels"
- schema "k8s.io/apimachinery/pkg/runtime/schema"
types "k8s.io/apimachinery/pkg/types"
watch "k8s.io/apimachinery/pkg/watch"
testing "k8s.io/client-go/testing"
@@ -36,9 +35,9 @@ type FakeServiceDescriptors struct {
ns string
}
-var servicedescriptorsResource = schema.GroupVersionResource{Group: "apps.kubeblocks.io", Version: "v1alpha1", Resource: "servicedescriptors"}
+var servicedescriptorsResource = v1alpha1.SchemeGroupVersion.WithResource("servicedescriptors")
-var servicedescriptorsKind = schema.GroupVersionKind{Group: "apps.kubeblocks.io", Version: "v1alpha1", Kind: "ServiceDescriptor"}
+var servicedescriptorsKind = v1alpha1.SchemeGroupVersion.WithKind("ServiceDescriptor")
// Get takes name of the serviceDescriptor, and returns the corresponding serviceDescriptor object, and an error if there is any.
func (c *FakeServiceDescriptors) Get(ctx context.Context, name string, options v1.GetOptions) (result *v1alpha1.ServiceDescriptor, err error) {
diff --git a/pkg/client/clientset/versioned/typed/dataprotection/v1alpha1/actionset.go b/pkg/client/clientset/versioned/typed/dataprotection/v1alpha1/actionset.go
new file mode 100644
index 00000000000..ef49d4eb84b
--- /dev/null
+++ b/pkg/client/clientset/versioned/typed/dataprotection/v1alpha1/actionset.go
@@ -0,0 +1,184 @@
+/*
+Copyright (C) 2022-2023 ApeCloud Co., Ltd
+
+Licensed under the Apache License, Version 2.0 (the "License");
+you may not use this file except in compliance with the License.
+You may obtain a copy of the License at
+
+ http://www.apache.org/licenses/LICENSE-2.0
+
+Unless required by applicable law or agreed to in writing, software
+distributed under the License is distributed on an "AS IS" BASIS,
+WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+See the License for the specific language governing permissions and
+limitations under the License.
+*/
+
+// Code generated by client-gen. DO NOT EDIT.
+
+package v1alpha1
+
+import (
+ "context"
+ "time"
+
+ v1alpha1 "github.com/apecloud/kubeblocks/apis/dataprotection/v1alpha1"
+ scheme "github.com/apecloud/kubeblocks/pkg/client/clientset/versioned/scheme"
+ v1 "k8s.io/apimachinery/pkg/apis/meta/v1"
+ types "k8s.io/apimachinery/pkg/types"
+ watch "k8s.io/apimachinery/pkg/watch"
+ rest "k8s.io/client-go/rest"
+)
+
+// ActionSetsGetter has a method to return a ActionSetInterface.
+// A group's client should implement this interface.
+type ActionSetsGetter interface {
+ ActionSets() ActionSetInterface
+}
+
+// ActionSetInterface has methods to work with ActionSet resources.
+type ActionSetInterface interface {
+ Create(ctx context.Context, actionSet *v1alpha1.ActionSet, opts v1.CreateOptions) (*v1alpha1.ActionSet, error)
+ Update(ctx context.Context, actionSet *v1alpha1.ActionSet, opts v1.UpdateOptions) (*v1alpha1.ActionSet, error)
+ UpdateStatus(ctx context.Context, actionSet *v1alpha1.ActionSet, opts v1.UpdateOptions) (*v1alpha1.ActionSet, error)
+ Delete(ctx context.Context, name string, opts v1.DeleteOptions) error
+ DeleteCollection(ctx context.Context, opts v1.DeleteOptions, listOpts v1.ListOptions) error
+ Get(ctx context.Context, name string, opts v1.GetOptions) (*v1alpha1.ActionSet, error)
+ List(ctx context.Context, opts v1.ListOptions) (*v1alpha1.ActionSetList, error)
+ Watch(ctx context.Context, opts v1.ListOptions) (watch.Interface, error)
+ Patch(ctx context.Context, name string, pt types.PatchType, data []byte, opts v1.PatchOptions, subresources ...string) (result *v1alpha1.ActionSet, err error)
+ ActionSetExpansion
+}
+
+// actionSets implements ActionSetInterface
+type actionSets struct {
+ client rest.Interface
+}
+
+// newActionSets returns a ActionSets
+func newActionSets(c *DataprotectionV1alpha1Client) *actionSets {
+ return &actionSets{
+ client: c.RESTClient(),
+ }
+}
+
+// Get takes name of the actionSet, and returns the corresponding actionSet object, and an error if there is any.
+func (c *actionSets) Get(ctx context.Context, name string, options v1.GetOptions) (result *v1alpha1.ActionSet, err error) {
+ result = &v1alpha1.ActionSet{}
+ err = c.client.Get().
+ Resource("actionsets").
+ Name(name).
+ VersionedParams(&options, scheme.ParameterCodec).
+ Do(ctx).
+ Into(result)
+ return
+}
+
+// List takes label and field selectors, and returns the list of ActionSets that match those selectors.
+func (c *actionSets) List(ctx context.Context, opts v1.ListOptions) (result *v1alpha1.ActionSetList, err error) {
+ var timeout time.Duration
+ if opts.TimeoutSeconds != nil {
+ timeout = time.Duration(*opts.TimeoutSeconds) * time.Second
+ }
+ result = &v1alpha1.ActionSetList{}
+ err = c.client.Get().
+ Resource("actionsets").
+ VersionedParams(&opts, scheme.ParameterCodec).
+ Timeout(timeout).
+ Do(ctx).
+ Into(result)
+ return
+}
+
+// Watch returns a watch.Interface that watches the requested actionSets.
+func (c *actionSets) Watch(ctx context.Context, opts v1.ListOptions) (watch.Interface, error) {
+ var timeout time.Duration
+ if opts.TimeoutSeconds != nil {
+ timeout = time.Duration(*opts.TimeoutSeconds) * time.Second
+ }
+ opts.Watch = true
+ return c.client.Get().
+ Resource("actionsets").
+ VersionedParams(&opts, scheme.ParameterCodec).
+ Timeout(timeout).
+ Watch(ctx)
+}
+
+// Create takes the representation of a actionSet and creates it. Returns the server's representation of the actionSet, and an error, if there is any.
+func (c *actionSets) Create(ctx context.Context, actionSet *v1alpha1.ActionSet, opts v1.CreateOptions) (result *v1alpha1.ActionSet, err error) {
+ result = &v1alpha1.ActionSet{}
+ err = c.client.Post().
+ Resource("actionsets").
+ VersionedParams(&opts, scheme.ParameterCodec).
+ Body(actionSet).
+ Do(ctx).
+ Into(result)
+ return
+}
+
+// Update takes the representation of a actionSet and updates it. Returns the server's representation of the actionSet, and an error, if there is any.
+func (c *actionSets) Update(ctx context.Context, actionSet *v1alpha1.ActionSet, opts v1.UpdateOptions) (result *v1alpha1.ActionSet, err error) {
+ result = &v1alpha1.ActionSet{}
+ err = c.client.Put().
+ Resource("actionsets").
+ Name(actionSet.Name).
+ VersionedParams(&opts, scheme.ParameterCodec).
+ Body(actionSet).
+ Do(ctx).
+ Into(result)
+ return
+}
+
+// UpdateStatus was generated because the type contains a Status member.
+// Add a +genclient:noStatus comment above the type to avoid generating UpdateStatus().
+func (c *actionSets) UpdateStatus(ctx context.Context, actionSet *v1alpha1.ActionSet, opts v1.UpdateOptions) (result *v1alpha1.ActionSet, err error) {
+ result = &v1alpha1.ActionSet{}
+ err = c.client.Put().
+ Resource("actionsets").
+ Name(actionSet.Name).
+ SubResource("status").
+ VersionedParams(&opts, scheme.ParameterCodec).
+ Body(actionSet).
+ Do(ctx).
+ Into(result)
+ return
+}
+
+// Delete takes name of the actionSet and deletes it. Returns an error if one occurs.
+func (c *actionSets) Delete(ctx context.Context, name string, opts v1.DeleteOptions) error {
+ return c.client.Delete().
+ Resource("actionsets").
+ Name(name).
+ Body(&opts).
+ Do(ctx).
+ Error()
+}
+
+// DeleteCollection deletes a collection of objects.
+func (c *actionSets) DeleteCollection(ctx context.Context, opts v1.DeleteOptions, listOpts v1.ListOptions) error {
+ var timeout time.Duration
+ if listOpts.TimeoutSeconds != nil {
+ timeout = time.Duration(*listOpts.TimeoutSeconds) * time.Second
+ }
+ return c.client.Delete().
+ Resource("actionsets").
+ VersionedParams(&listOpts, scheme.ParameterCodec).
+ Timeout(timeout).
+ Body(&opts).
+ Do(ctx).
+ Error()
+}
+
+// Patch applies the patch and returns the patched actionSet.
+func (c *actionSets) Patch(ctx context.Context, name string, pt types.PatchType, data []byte, opts v1.PatchOptions, subresources ...string) (result *v1alpha1.ActionSet, err error) {
+ result = &v1alpha1.ActionSet{}
+ err = c.client.Patch(pt).
+ Resource("actionsets").
+ Name(name).
+ SubResource(subresources...).
+ VersionedParams(&opts, scheme.ParameterCodec).
+ Body(data).
+ Do(ctx).
+ Into(result)
+ return
+}
diff --git a/pkg/client/clientset/versioned/typed/dataprotection/v1alpha1/backupschedule.go b/pkg/client/clientset/versioned/typed/dataprotection/v1alpha1/backupschedule.go
new file mode 100644
index 00000000000..5094b4a3a9b
--- /dev/null
+++ b/pkg/client/clientset/versioned/typed/dataprotection/v1alpha1/backupschedule.go
@@ -0,0 +1,195 @@
+/*
+Copyright (C) 2022-2023 ApeCloud Co., Ltd
+
+Licensed under the Apache License, Version 2.0 (the "License");
+you may not use this file except in compliance with the License.
+You may obtain a copy of the License at
+
+ http://www.apache.org/licenses/LICENSE-2.0
+
+Unless required by applicable law or agreed to in writing, software
+distributed under the License is distributed on an "AS IS" BASIS,
+WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+See the License for the specific language governing permissions and
+limitations under the License.
+*/
+
+// Code generated by client-gen. DO NOT EDIT.
+
+package v1alpha1
+
+import (
+ "context"
+ "time"
+
+ v1alpha1 "github.com/apecloud/kubeblocks/apis/dataprotection/v1alpha1"
+ scheme "github.com/apecloud/kubeblocks/pkg/client/clientset/versioned/scheme"
+ v1 "k8s.io/apimachinery/pkg/apis/meta/v1"
+ types "k8s.io/apimachinery/pkg/types"
+ watch "k8s.io/apimachinery/pkg/watch"
+ rest "k8s.io/client-go/rest"
+)
+
+// BackupSchedulesGetter has a method to return a BackupScheduleInterface.
+// A group's client should implement this interface.
+type BackupSchedulesGetter interface {
+ BackupSchedules(namespace string) BackupScheduleInterface
+}
+
+// BackupScheduleInterface has methods to work with BackupSchedule resources.
+type BackupScheduleInterface interface {
+ Create(ctx context.Context, backupSchedule *v1alpha1.BackupSchedule, opts v1.CreateOptions) (*v1alpha1.BackupSchedule, error)
+ Update(ctx context.Context, backupSchedule *v1alpha1.BackupSchedule, opts v1.UpdateOptions) (*v1alpha1.BackupSchedule, error)
+ UpdateStatus(ctx context.Context, backupSchedule *v1alpha1.BackupSchedule, opts v1.UpdateOptions) (*v1alpha1.BackupSchedule, error)
+ Delete(ctx context.Context, name string, opts v1.DeleteOptions) error
+ DeleteCollection(ctx context.Context, opts v1.DeleteOptions, listOpts v1.ListOptions) error
+ Get(ctx context.Context, name string, opts v1.GetOptions) (*v1alpha1.BackupSchedule, error)
+ List(ctx context.Context, opts v1.ListOptions) (*v1alpha1.BackupScheduleList, error)
+ Watch(ctx context.Context, opts v1.ListOptions) (watch.Interface, error)
+ Patch(ctx context.Context, name string, pt types.PatchType, data []byte, opts v1.PatchOptions, subresources ...string) (result *v1alpha1.BackupSchedule, err error)
+ BackupScheduleExpansion
+}
+
+// backupSchedules implements BackupScheduleInterface
+type backupSchedules struct {
+ client rest.Interface
+ ns string
+}
+
+// newBackupSchedules returns a BackupSchedules
+func newBackupSchedules(c *DataprotectionV1alpha1Client, namespace string) *backupSchedules {
+ return &backupSchedules{
+ client: c.RESTClient(),
+ ns: namespace,
+ }
+}
+
+// Get takes name of the backupSchedule, and returns the corresponding backupSchedule object, and an error if there is any.
+func (c *backupSchedules) Get(ctx context.Context, name string, options v1.GetOptions) (result *v1alpha1.BackupSchedule, err error) {
+ result = &v1alpha1.BackupSchedule{}
+ err = c.client.Get().
+ Namespace(c.ns).
+ Resource("backupschedules").
+ Name(name).
+ VersionedParams(&options, scheme.ParameterCodec).
+ Do(ctx).
+ Into(result)
+ return
+}
+
+// List takes label and field selectors, and returns the list of BackupSchedules that match those selectors.
+func (c *backupSchedules) List(ctx context.Context, opts v1.ListOptions) (result *v1alpha1.BackupScheduleList, err error) {
+ var timeout time.Duration
+ if opts.TimeoutSeconds != nil {
+ timeout = time.Duration(*opts.TimeoutSeconds) * time.Second
+ }
+ result = &v1alpha1.BackupScheduleList{}
+ err = c.client.Get().
+ Namespace(c.ns).
+ Resource("backupschedules").
+ VersionedParams(&opts, scheme.ParameterCodec).
+ Timeout(timeout).
+ Do(ctx).
+ Into(result)
+ return
+}
+
+// Watch returns a watch.Interface that watches the requested backupSchedules.
+func (c *backupSchedules) Watch(ctx context.Context, opts v1.ListOptions) (watch.Interface, error) {
+ var timeout time.Duration
+ if opts.TimeoutSeconds != nil {
+ timeout = time.Duration(*opts.TimeoutSeconds) * time.Second
+ }
+ opts.Watch = true
+ return c.client.Get().
+ Namespace(c.ns).
+ Resource("backupschedules").
+ VersionedParams(&opts, scheme.ParameterCodec).
+ Timeout(timeout).
+ Watch(ctx)
+}
+
+// Create takes the representation of a backupSchedule and creates it. Returns the server's representation of the backupSchedule, and an error, if there is any.
+func (c *backupSchedules) Create(ctx context.Context, backupSchedule *v1alpha1.BackupSchedule, opts v1.CreateOptions) (result *v1alpha1.BackupSchedule, err error) {
+ result = &v1alpha1.BackupSchedule{}
+ err = c.client.Post().
+ Namespace(c.ns).
+ Resource("backupschedules").
+ VersionedParams(&opts, scheme.ParameterCodec).
+ Body(backupSchedule).
+ Do(ctx).
+ Into(result)
+ return
+}
+
+// Update takes the representation of a backupSchedule and updates it. Returns the server's representation of the backupSchedule, and an error, if there is any.
+func (c *backupSchedules) Update(ctx context.Context, backupSchedule *v1alpha1.BackupSchedule, opts v1.UpdateOptions) (result *v1alpha1.BackupSchedule, err error) {
+ result = &v1alpha1.BackupSchedule{}
+ err = c.client.Put().
+ Namespace(c.ns).
+ Resource("backupschedules").
+ Name(backupSchedule.Name).
+ VersionedParams(&opts, scheme.ParameterCodec).
+ Body(backupSchedule).
+ Do(ctx).
+ Into(result)
+ return
+}
+
+// UpdateStatus was generated because the type contains a Status member.
+// Add a +genclient:noStatus comment above the type to avoid generating UpdateStatus().
+func (c *backupSchedules) UpdateStatus(ctx context.Context, backupSchedule *v1alpha1.BackupSchedule, opts v1.UpdateOptions) (result *v1alpha1.BackupSchedule, err error) {
+ result = &v1alpha1.BackupSchedule{}
+ err = c.client.Put().
+ Namespace(c.ns).
+ Resource("backupschedules").
+ Name(backupSchedule.Name).
+ SubResource("status").
+ VersionedParams(&opts, scheme.ParameterCodec).
+ Body(backupSchedule).
+ Do(ctx).
+ Into(result)
+ return
+}
+
+// Delete takes name of the backupSchedule and deletes it. Returns an error if one occurs.
+func (c *backupSchedules) Delete(ctx context.Context, name string, opts v1.DeleteOptions) error {
+ return c.client.Delete().
+ Namespace(c.ns).
+ Resource("backupschedules").
+ Name(name).
+ Body(&opts).
+ Do(ctx).
+ Error()
+}
+
+// DeleteCollection deletes a collection of objects.
+func (c *backupSchedules) DeleteCollection(ctx context.Context, opts v1.DeleteOptions, listOpts v1.ListOptions) error {
+ var timeout time.Duration
+ if listOpts.TimeoutSeconds != nil {
+ timeout = time.Duration(*listOpts.TimeoutSeconds) * time.Second
+ }
+ return c.client.Delete().
+ Namespace(c.ns).
+ Resource("backupschedules").
+ VersionedParams(&listOpts, scheme.ParameterCodec).
+ Timeout(timeout).
+ Body(&opts).
+ Do(ctx).
+ Error()
+}
+
+// Patch applies the patch and returns the patched backupSchedule.
+func (c *backupSchedules) Patch(ctx context.Context, name string, pt types.PatchType, data []byte, opts v1.PatchOptions, subresources ...string) (result *v1alpha1.BackupSchedule, err error) {
+ result = &v1alpha1.BackupSchedule{}
+ err = c.client.Patch(pt).
+ Namespace(c.ns).
+ Resource("backupschedules").
+ Name(name).
+ SubResource(subresources...).
+ VersionedParams(&opts, scheme.ParameterCodec).
+ Body(data).
+ Do(ctx).
+ Into(result)
+ return
+}
diff --git a/pkg/client/clientset/versioned/typed/dataprotection/v1alpha1/backuptool.go b/pkg/client/clientset/versioned/typed/dataprotection/v1alpha1/backuptool.go
deleted file mode 100644
index 2362bf20450..00000000000
--- a/pkg/client/clientset/versioned/typed/dataprotection/v1alpha1/backuptool.go
+++ /dev/null
@@ -1,184 +0,0 @@
-/*
-Copyright (C) 2022-2023 ApeCloud Co., Ltd
-
-Licensed under the Apache License, Version 2.0 (the "License");
-you may not use this file except in compliance with the License.
-You may obtain a copy of the License at
-
- http://www.apache.org/licenses/LICENSE-2.0
-
-Unless required by applicable law or agreed to in writing, software
-distributed under the License is distributed on an "AS IS" BASIS,
-WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
-See the License for the specific language governing permissions and
-limitations under the License.
-*/
-
-// Code generated by client-gen. DO NOT EDIT.
-
-package v1alpha1
-
-import (
- "context"
- "time"
-
- v1alpha1 "github.com/apecloud/kubeblocks/apis/dataprotection/v1alpha1"
- scheme "github.com/apecloud/kubeblocks/pkg/client/clientset/versioned/scheme"
- v1 "k8s.io/apimachinery/pkg/apis/meta/v1"
- types "k8s.io/apimachinery/pkg/types"
- watch "k8s.io/apimachinery/pkg/watch"
- rest "k8s.io/client-go/rest"
-)
-
-// BackupToolsGetter has a method to return a BackupToolInterface.
-// A group's client should implement this interface.
-type BackupToolsGetter interface {
- BackupTools() BackupToolInterface
-}
-
-// BackupToolInterface has methods to work with BackupTool resources.
-type BackupToolInterface interface {
- Create(ctx context.Context, backupTool *v1alpha1.BackupTool, opts v1.CreateOptions) (*v1alpha1.BackupTool, error)
- Update(ctx context.Context, backupTool *v1alpha1.BackupTool, opts v1.UpdateOptions) (*v1alpha1.BackupTool, error)
- UpdateStatus(ctx context.Context, backupTool *v1alpha1.BackupTool, opts v1.UpdateOptions) (*v1alpha1.BackupTool, error)
- Delete(ctx context.Context, name string, opts v1.DeleteOptions) error
- DeleteCollection(ctx context.Context, opts v1.DeleteOptions, listOpts v1.ListOptions) error
- Get(ctx context.Context, name string, opts v1.GetOptions) (*v1alpha1.BackupTool, error)
- List(ctx context.Context, opts v1.ListOptions) (*v1alpha1.BackupToolList, error)
- Watch(ctx context.Context, opts v1.ListOptions) (watch.Interface, error)
- Patch(ctx context.Context, name string, pt types.PatchType, data []byte, opts v1.PatchOptions, subresources ...string) (result *v1alpha1.BackupTool, err error)
- BackupToolExpansion
-}
-
-// backupTools implements BackupToolInterface
-type backupTools struct {
- client rest.Interface
-}
-
-// newBackupTools returns a BackupTools
-func newBackupTools(c *DataprotectionV1alpha1Client) *backupTools {
- return &backupTools{
- client: c.RESTClient(),
- }
-}
-
-// Get takes name of the backupTool, and returns the corresponding backupTool object, and an error if there is any.
-func (c *backupTools) Get(ctx context.Context, name string, options v1.GetOptions) (result *v1alpha1.BackupTool, err error) {
- result = &v1alpha1.BackupTool{}
- err = c.client.Get().
- Resource("backuptools").
- Name(name).
- VersionedParams(&options, scheme.ParameterCodec).
- Do(ctx).
- Into(result)
- return
-}
-
-// List takes label and field selectors, and returns the list of BackupTools that match those selectors.
-func (c *backupTools) List(ctx context.Context, opts v1.ListOptions) (result *v1alpha1.BackupToolList, err error) {
- var timeout time.Duration
- if opts.TimeoutSeconds != nil {
- timeout = time.Duration(*opts.TimeoutSeconds) * time.Second
- }
- result = &v1alpha1.BackupToolList{}
- err = c.client.Get().
- Resource("backuptools").
- VersionedParams(&opts, scheme.ParameterCodec).
- Timeout(timeout).
- Do(ctx).
- Into(result)
- return
-}
-
-// Watch returns a watch.Interface that watches the requested backupTools.
-func (c *backupTools) Watch(ctx context.Context, opts v1.ListOptions) (watch.Interface, error) {
- var timeout time.Duration
- if opts.TimeoutSeconds != nil {
- timeout = time.Duration(*opts.TimeoutSeconds) * time.Second
- }
- opts.Watch = true
- return c.client.Get().
- Resource("backuptools").
- VersionedParams(&opts, scheme.ParameterCodec).
- Timeout(timeout).
- Watch(ctx)
-}
-
-// Create takes the representation of a backupTool and creates it. Returns the server's representation of the backupTool, and an error, if there is any.
-func (c *backupTools) Create(ctx context.Context, backupTool *v1alpha1.BackupTool, opts v1.CreateOptions) (result *v1alpha1.BackupTool, err error) {
- result = &v1alpha1.BackupTool{}
- err = c.client.Post().
- Resource("backuptools").
- VersionedParams(&opts, scheme.ParameterCodec).
- Body(backupTool).
- Do(ctx).
- Into(result)
- return
-}
-
-// Update takes the representation of a backupTool and updates it. Returns the server's representation of the backupTool, and an error, if there is any.
-func (c *backupTools) Update(ctx context.Context, backupTool *v1alpha1.BackupTool, opts v1.UpdateOptions) (result *v1alpha1.BackupTool, err error) {
- result = &v1alpha1.BackupTool{}
- err = c.client.Put().
- Resource("backuptools").
- Name(backupTool.Name).
- VersionedParams(&opts, scheme.ParameterCodec).
- Body(backupTool).
- Do(ctx).
- Into(result)
- return
-}
-
-// UpdateStatus was generated because the type contains a Status member.
-// Add a +genclient:noStatus comment above the type to avoid generating UpdateStatus().
-func (c *backupTools) UpdateStatus(ctx context.Context, backupTool *v1alpha1.BackupTool, opts v1.UpdateOptions) (result *v1alpha1.BackupTool, err error) {
- result = &v1alpha1.BackupTool{}
- err = c.client.Put().
- Resource("backuptools").
- Name(backupTool.Name).
- SubResource("status").
- VersionedParams(&opts, scheme.ParameterCodec).
- Body(backupTool).
- Do(ctx).
- Into(result)
- return
-}
-
-// Delete takes name of the backupTool and deletes it. Returns an error if one occurs.
-func (c *backupTools) Delete(ctx context.Context, name string, opts v1.DeleteOptions) error {
- return c.client.Delete().
- Resource("backuptools").
- Name(name).
- Body(&opts).
- Do(ctx).
- Error()
-}
-
-// DeleteCollection deletes a collection of objects.
-func (c *backupTools) DeleteCollection(ctx context.Context, opts v1.DeleteOptions, listOpts v1.ListOptions) error {
- var timeout time.Duration
- if listOpts.TimeoutSeconds != nil {
- timeout = time.Duration(*listOpts.TimeoutSeconds) * time.Second
- }
- return c.client.Delete().
- Resource("backuptools").
- VersionedParams(&listOpts, scheme.ParameterCodec).
- Timeout(timeout).
- Body(&opts).
- Do(ctx).
- Error()
-}
-
-// Patch applies the patch and returns the patched backupTool.
-func (c *backupTools) Patch(ctx context.Context, name string, pt types.PatchType, data []byte, opts v1.PatchOptions, subresources ...string) (result *v1alpha1.BackupTool, err error) {
- result = &v1alpha1.BackupTool{}
- err = c.client.Patch(pt).
- Resource("backuptools").
- Name(name).
- SubResource(subresources...).
- VersionedParams(&opts, scheme.ParameterCodec).
- Body(data).
- Do(ctx).
- Into(result)
- return
-}
diff --git a/pkg/client/clientset/versioned/typed/dataprotection/v1alpha1/dataprotection_client.go b/pkg/client/clientset/versioned/typed/dataprotection/v1alpha1/dataprotection_client.go
index 37e95aeff31..54579c00671 100644
--- a/pkg/client/clientset/versioned/typed/dataprotection/v1alpha1/dataprotection_client.go
+++ b/pkg/client/clientset/versioned/typed/dataprotection/v1alpha1/dataprotection_client.go
@@ -28,11 +28,12 @@ import (
type DataprotectionV1alpha1Interface interface {
RESTClient() rest.Interface
+ ActionSetsGetter
BackupsGetter
BackupPoliciesGetter
BackupReposGetter
- BackupToolsGetter
- RestoreJobsGetter
+ BackupSchedulesGetter
+ RestoresGetter
}
// DataprotectionV1alpha1Client is used to interact with features provided by the dataprotection.kubeblocks.io group.
@@ -40,6 +41,10 @@ type DataprotectionV1alpha1Client struct {
restClient rest.Interface
}
+func (c *DataprotectionV1alpha1Client) ActionSets() ActionSetInterface {
+ return newActionSets(c)
+}
+
func (c *DataprotectionV1alpha1Client) Backups(namespace string) BackupInterface {
return newBackups(c, namespace)
}
@@ -52,12 +57,12 @@ func (c *DataprotectionV1alpha1Client) BackupRepos() BackupRepoInterface {
return newBackupRepos(c)
}
-func (c *DataprotectionV1alpha1Client) BackupTools() BackupToolInterface {
- return newBackupTools(c)
+func (c *DataprotectionV1alpha1Client) BackupSchedules(namespace string) BackupScheduleInterface {
+ return newBackupSchedules(c, namespace)
}
-func (c *DataprotectionV1alpha1Client) RestoreJobs(namespace string) RestoreJobInterface {
- return newRestoreJobs(c, namespace)
+func (c *DataprotectionV1alpha1Client) Restores(namespace string) RestoreInterface {
+ return newRestores(c, namespace)
}
// NewForConfig creates a new DataprotectionV1alpha1Client for the given config.
diff --git a/pkg/client/clientset/versioned/typed/dataprotection/v1alpha1/fake/fake_actionset.go b/pkg/client/clientset/versioned/typed/dataprotection/v1alpha1/fake/fake_actionset.go
new file mode 100644
index 00000000000..2ffd828b4ff
--- /dev/null
+++ b/pkg/client/clientset/versioned/typed/dataprotection/v1alpha1/fake/fake_actionset.go
@@ -0,0 +1,132 @@
+/*
+Copyright (C) 2022-2023 ApeCloud Co., Ltd
+
+Licensed under the Apache License, Version 2.0 (the "License");
+you may not use this file except in compliance with the License.
+You may obtain a copy of the License at
+
+ http://www.apache.org/licenses/LICENSE-2.0
+
+Unless required by applicable law or agreed to in writing, software
+distributed under the License is distributed on an "AS IS" BASIS,
+WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+See the License for the specific language governing permissions and
+limitations under the License.
+*/
+
+// Code generated by client-gen. DO NOT EDIT.
+
+package fake
+
+import (
+ "context"
+
+ v1alpha1 "github.com/apecloud/kubeblocks/apis/dataprotection/v1alpha1"
+ v1 "k8s.io/apimachinery/pkg/apis/meta/v1"
+ labels "k8s.io/apimachinery/pkg/labels"
+ types "k8s.io/apimachinery/pkg/types"
+ watch "k8s.io/apimachinery/pkg/watch"
+ testing "k8s.io/client-go/testing"
+)
+
+// FakeActionSets implements ActionSetInterface
+type FakeActionSets struct {
+ Fake *FakeDataprotectionV1alpha1
+}
+
+var actionsetsResource = v1alpha1.SchemeGroupVersion.WithResource("actionsets")
+
+var actionsetsKind = v1alpha1.SchemeGroupVersion.WithKind("ActionSet")
+
+// Get takes name of the actionSet, and returns the corresponding actionSet object, and an error if there is any.
+func (c *FakeActionSets) Get(ctx context.Context, name string, options v1.GetOptions) (result *v1alpha1.ActionSet, err error) {
+ obj, err := c.Fake.
+ Invokes(testing.NewRootGetAction(actionsetsResource, name), &v1alpha1.ActionSet{})
+ if obj == nil {
+ return nil, err
+ }
+ return obj.(*v1alpha1.ActionSet), err
+}
+
+// List takes label and field selectors, and returns the list of ActionSets that match those selectors.
+func (c *FakeActionSets) List(ctx context.Context, opts v1.ListOptions) (result *v1alpha1.ActionSetList, err error) {
+ obj, err := c.Fake.
+ Invokes(testing.NewRootListAction(actionsetsResource, actionsetsKind, opts), &v1alpha1.ActionSetList{})
+ if obj == nil {
+ return nil, err
+ }
+
+ label, _, _ := testing.ExtractFromListOptions(opts)
+ if label == nil {
+ label = labels.Everything()
+ }
+ list := &v1alpha1.ActionSetList{ListMeta: obj.(*v1alpha1.ActionSetList).ListMeta}
+ for _, item := range obj.(*v1alpha1.ActionSetList).Items {
+ if label.Matches(labels.Set(item.Labels)) {
+ list.Items = append(list.Items, item)
+ }
+ }
+ return list, err
+}
+
+// Watch returns a watch.Interface that watches the requested actionSets.
+func (c *FakeActionSets) Watch(ctx context.Context, opts v1.ListOptions) (watch.Interface, error) {
+ return c.Fake.
+ InvokesWatch(testing.NewRootWatchAction(actionsetsResource, opts))
+}
+
+// Create takes the representation of a actionSet and creates it. Returns the server's representation of the actionSet, and an error, if there is any.
+func (c *FakeActionSets) Create(ctx context.Context, actionSet *v1alpha1.ActionSet, opts v1.CreateOptions) (result *v1alpha1.ActionSet, err error) {
+ obj, err := c.Fake.
+ Invokes(testing.NewRootCreateAction(actionsetsResource, actionSet), &v1alpha1.ActionSet{})
+ if obj == nil {
+ return nil, err
+ }
+ return obj.(*v1alpha1.ActionSet), err
+}
+
+// Update takes the representation of a actionSet and updates it. Returns the server's representation of the actionSet, and an error, if there is any.
+func (c *FakeActionSets) Update(ctx context.Context, actionSet *v1alpha1.ActionSet, opts v1.UpdateOptions) (result *v1alpha1.ActionSet, err error) {
+ obj, err := c.Fake.
+ Invokes(testing.NewRootUpdateAction(actionsetsResource, actionSet), &v1alpha1.ActionSet{})
+ if obj == nil {
+ return nil, err
+ }
+ return obj.(*v1alpha1.ActionSet), err
+}
+
+// UpdateStatus was generated because the type contains a Status member.
+// Add a +genclient:noStatus comment above the type to avoid generating UpdateStatus().
+func (c *FakeActionSets) UpdateStatus(ctx context.Context, actionSet *v1alpha1.ActionSet, opts v1.UpdateOptions) (*v1alpha1.ActionSet, error) {
+ obj, err := c.Fake.
+ Invokes(testing.NewRootUpdateSubresourceAction(actionsetsResource, "status", actionSet), &v1alpha1.ActionSet{})
+ if obj == nil {
+ return nil, err
+ }
+ return obj.(*v1alpha1.ActionSet), err
+}
+
+// Delete takes name of the actionSet and deletes it. Returns an error if one occurs.
+func (c *FakeActionSets) Delete(ctx context.Context, name string, opts v1.DeleteOptions) error {
+ _, err := c.Fake.
+ Invokes(testing.NewRootDeleteActionWithOptions(actionsetsResource, name, opts), &v1alpha1.ActionSet{})
+ return err
+}
+
+// DeleteCollection deletes a collection of objects.
+func (c *FakeActionSets) DeleteCollection(ctx context.Context, opts v1.DeleteOptions, listOpts v1.ListOptions) error {
+ action := testing.NewRootDeleteCollectionAction(actionsetsResource, listOpts)
+
+ _, err := c.Fake.Invokes(action, &v1alpha1.ActionSetList{})
+ return err
+}
+
+// Patch applies the patch and returns the patched actionSet.
+func (c *FakeActionSets) Patch(ctx context.Context, name string, pt types.PatchType, data []byte, opts v1.PatchOptions, subresources ...string) (result *v1alpha1.ActionSet, err error) {
+ obj, err := c.Fake.
+ Invokes(testing.NewRootPatchSubresourceAction(actionsetsResource, name, pt, data, subresources...), &v1alpha1.ActionSet{})
+ if obj == nil {
+ return nil, err
+ }
+ return obj.(*v1alpha1.ActionSet), err
+}
diff --git a/pkg/client/clientset/versioned/typed/dataprotection/v1alpha1/fake/fake_backup.go b/pkg/client/clientset/versioned/typed/dataprotection/v1alpha1/fake/fake_backup.go
index d0df0d9e38d..13b05e5e82c 100644
--- a/pkg/client/clientset/versioned/typed/dataprotection/v1alpha1/fake/fake_backup.go
+++ b/pkg/client/clientset/versioned/typed/dataprotection/v1alpha1/fake/fake_backup.go
@@ -24,7 +24,6 @@ import (
v1alpha1 "github.com/apecloud/kubeblocks/apis/dataprotection/v1alpha1"
v1 "k8s.io/apimachinery/pkg/apis/meta/v1"
labels "k8s.io/apimachinery/pkg/labels"
- schema "k8s.io/apimachinery/pkg/runtime/schema"
types "k8s.io/apimachinery/pkg/types"
watch "k8s.io/apimachinery/pkg/watch"
testing "k8s.io/client-go/testing"
@@ -36,9 +35,9 @@ type FakeBackups struct {
ns string
}
-var backupsResource = schema.GroupVersionResource{Group: "dataprotection.kubeblocks.io", Version: "v1alpha1", Resource: "backups"}
+var backupsResource = v1alpha1.SchemeGroupVersion.WithResource("backups")
-var backupsKind = schema.GroupVersionKind{Group: "dataprotection.kubeblocks.io", Version: "v1alpha1", Kind: "Backup"}
+var backupsKind = v1alpha1.SchemeGroupVersion.WithKind("Backup")
// Get takes name of the backup, and returns the corresponding backup object, and an error if there is any.
func (c *FakeBackups) Get(ctx context.Context, name string, options v1.GetOptions) (result *v1alpha1.Backup, err error) {
diff --git a/pkg/client/clientset/versioned/typed/dataprotection/v1alpha1/fake/fake_backuppolicy.go b/pkg/client/clientset/versioned/typed/dataprotection/v1alpha1/fake/fake_backuppolicy.go
index 75267050936..68b567006fa 100644
--- a/pkg/client/clientset/versioned/typed/dataprotection/v1alpha1/fake/fake_backuppolicy.go
+++ b/pkg/client/clientset/versioned/typed/dataprotection/v1alpha1/fake/fake_backuppolicy.go
@@ -24,7 +24,6 @@ import (
v1alpha1 "github.com/apecloud/kubeblocks/apis/dataprotection/v1alpha1"
v1 "k8s.io/apimachinery/pkg/apis/meta/v1"
labels "k8s.io/apimachinery/pkg/labels"
- schema "k8s.io/apimachinery/pkg/runtime/schema"
types "k8s.io/apimachinery/pkg/types"
watch "k8s.io/apimachinery/pkg/watch"
testing "k8s.io/client-go/testing"
@@ -36,9 +35,9 @@ type FakeBackupPolicies struct {
ns string
}
-var backuppoliciesResource = schema.GroupVersionResource{Group: "dataprotection.kubeblocks.io", Version: "v1alpha1", Resource: "backuppolicies"}
+var backuppoliciesResource = v1alpha1.SchemeGroupVersion.WithResource("backuppolicies")
-var backuppoliciesKind = schema.GroupVersionKind{Group: "dataprotection.kubeblocks.io", Version: "v1alpha1", Kind: "BackupPolicy"}
+var backuppoliciesKind = v1alpha1.SchemeGroupVersion.WithKind("BackupPolicy")
// Get takes name of the backupPolicy, and returns the corresponding backupPolicy object, and an error if there is any.
func (c *FakeBackupPolicies) Get(ctx context.Context, name string, options v1.GetOptions) (result *v1alpha1.BackupPolicy, err error) {
diff --git a/pkg/client/clientset/versioned/typed/dataprotection/v1alpha1/fake/fake_backuprepo.go b/pkg/client/clientset/versioned/typed/dataprotection/v1alpha1/fake/fake_backuprepo.go
index 6d71ea05822..37f77268ca2 100644
--- a/pkg/client/clientset/versioned/typed/dataprotection/v1alpha1/fake/fake_backuprepo.go
+++ b/pkg/client/clientset/versioned/typed/dataprotection/v1alpha1/fake/fake_backuprepo.go
@@ -24,7 +24,6 @@ import (
v1alpha1 "github.com/apecloud/kubeblocks/apis/dataprotection/v1alpha1"
v1 "k8s.io/apimachinery/pkg/apis/meta/v1"
labels "k8s.io/apimachinery/pkg/labels"
- schema "k8s.io/apimachinery/pkg/runtime/schema"
types "k8s.io/apimachinery/pkg/types"
watch "k8s.io/apimachinery/pkg/watch"
testing "k8s.io/client-go/testing"
@@ -35,9 +34,9 @@ type FakeBackupRepos struct {
Fake *FakeDataprotectionV1alpha1
}
-var backupreposResource = schema.GroupVersionResource{Group: "dataprotection.kubeblocks.io", Version: "v1alpha1", Resource: "backuprepos"}
+var backupreposResource = v1alpha1.SchemeGroupVersion.WithResource("backuprepos")
-var backupreposKind = schema.GroupVersionKind{Group: "dataprotection.kubeblocks.io", Version: "v1alpha1", Kind: "BackupRepo"}
+var backupreposKind = v1alpha1.SchemeGroupVersion.WithKind("BackupRepo")
// Get takes name of the backupRepo, and returns the corresponding backupRepo object, and an error if there is any.
func (c *FakeBackupRepos) Get(ctx context.Context, name string, options v1.GetOptions) (result *v1alpha1.BackupRepo, err error) {
diff --git a/pkg/client/clientset/versioned/typed/dataprotection/v1alpha1/fake/fake_backupschedule.go b/pkg/client/clientset/versioned/typed/dataprotection/v1alpha1/fake/fake_backupschedule.go
new file mode 100644
index 00000000000..42f6bfd51c7
--- /dev/null
+++ b/pkg/client/clientset/versioned/typed/dataprotection/v1alpha1/fake/fake_backupschedule.go
@@ -0,0 +1,141 @@
+/*
+Copyright (C) 2022-2023 ApeCloud Co., Ltd
+
+Licensed under the Apache License, Version 2.0 (the "License");
+you may not use this file except in compliance with the License.
+You may obtain a copy of the License at
+
+ http://www.apache.org/licenses/LICENSE-2.0
+
+Unless required by applicable law or agreed to in writing, software
+distributed under the License is distributed on an "AS IS" BASIS,
+WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+See the License for the specific language governing permissions and
+limitations under the License.
+*/
+
+// Code generated by client-gen. DO NOT EDIT.
+
+package fake
+
+import (
+ "context"
+
+ v1alpha1 "github.com/apecloud/kubeblocks/apis/dataprotection/v1alpha1"
+ v1 "k8s.io/apimachinery/pkg/apis/meta/v1"
+ labels "k8s.io/apimachinery/pkg/labels"
+ types "k8s.io/apimachinery/pkg/types"
+ watch "k8s.io/apimachinery/pkg/watch"
+ testing "k8s.io/client-go/testing"
+)
+
+// FakeBackupSchedules implements BackupScheduleInterface
+type FakeBackupSchedules struct {
+ Fake *FakeDataprotectionV1alpha1
+ ns string
+}
+
+var backupschedulesResource = v1alpha1.SchemeGroupVersion.WithResource("backupschedules")
+
+var backupschedulesKind = v1alpha1.SchemeGroupVersion.WithKind("BackupSchedule")
+
+// Get takes name of the backupSchedule, and returns the corresponding backupSchedule object, and an error if there is any.
+func (c *FakeBackupSchedules) Get(ctx context.Context, name string, options v1.GetOptions) (result *v1alpha1.BackupSchedule, err error) {
+ obj, err := c.Fake.
+ Invokes(testing.NewGetAction(backupschedulesResource, c.ns, name), &v1alpha1.BackupSchedule{})
+
+ if obj == nil {
+ return nil, err
+ }
+ return obj.(*v1alpha1.BackupSchedule), err
+}
+
+// List takes label and field selectors, and returns the list of BackupSchedules that match those selectors.
+func (c *FakeBackupSchedules) List(ctx context.Context, opts v1.ListOptions) (result *v1alpha1.BackupScheduleList, err error) {
+ obj, err := c.Fake.
+ Invokes(testing.NewListAction(backupschedulesResource, backupschedulesKind, c.ns, opts), &v1alpha1.BackupScheduleList{})
+
+ if obj == nil {
+ return nil, err
+ }
+
+ label, _, _ := testing.ExtractFromListOptions(opts)
+ if label == nil {
+ label = labels.Everything()
+ }
+ list := &v1alpha1.BackupScheduleList{ListMeta: obj.(*v1alpha1.BackupScheduleList).ListMeta}
+ for _, item := range obj.(*v1alpha1.BackupScheduleList).Items {
+ if label.Matches(labels.Set(item.Labels)) {
+ list.Items = append(list.Items, item)
+ }
+ }
+ return list, err
+}
+
+// Watch returns a watch.Interface that watches the requested backupSchedules.
+func (c *FakeBackupSchedules) Watch(ctx context.Context, opts v1.ListOptions) (watch.Interface, error) {
+ return c.Fake.
+ InvokesWatch(testing.NewWatchAction(backupschedulesResource, c.ns, opts))
+
+}
+
+// Create takes the representation of a backupSchedule and creates it. Returns the server's representation of the backupSchedule, and an error, if there is any.
+func (c *FakeBackupSchedules) Create(ctx context.Context, backupSchedule *v1alpha1.BackupSchedule, opts v1.CreateOptions) (result *v1alpha1.BackupSchedule, err error) {
+ obj, err := c.Fake.
+ Invokes(testing.NewCreateAction(backupschedulesResource, c.ns, backupSchedule), &v1alpha1.BackupSchedule{})
+
+ if obj == nil {
+ return nil, err
+ }
+ return obj.(*v1alpha1.BackupSchedule), err
+}
+
+// Update takes the representation of a backupSchedule and updates it. Returns the server's representation of the backupSchedule, and an error, if there is any.
+func (c *FakeBackupSchedules) Update(ctx context.Context, backupSchedule *v1alpha1.BackupSchedule, opts v1.UpdateOptions) (result *v1alpha1.BackupSchedule, err error) {
+ obj, err := c.Fake.
+ Invokes(testing.NewUpdateAction(backupschedulesResource, c.ns, backupSchedule), &v1alpha1.BackupSchedule{})
+
+ if obj == nil {
+ return nil, err
+ }
+ return obj.(*v1alpha1.BackupSchedule), err
+}
+
+// UpdateStatus was generated because the type contains a Status member.
+// Add a +genclient:noStatus comment above the type to avoid generating UpdateStatus().
+func (c *FakeBackupSchedules) UpdateStatus(ctx context.Context, backupSchedule *v1alpha1.BackupSchedule, opts v1.UpdateOptions) (*v1alpha1.BackupSchedule, error) {
+ obj, err := c.Fake.
+ Invokes(testing.NewUpdateSubresourceAction(backupschedulesResource, "status", c.ns, backupSchedule), &v1alpha1.BackupSchedule{})
+
+ if obj == nil {
+ return nil, err
+ }
+ return obj.(*v1alpha1.BackupSchedule), err
+}
+
+// Delete takes name of the backupSchedule and deletes it. Returns an error if one occurs.
+func (c *FakeBackupSchedules) Delete(ctx context.Context, name string, opts v1.DeleteOptions) error {
+ _, err := c.Fake.
+ Invokes(testing.NewDeleteActionWithOptions(backupschedulesResource, c.ns, name, opts), &v1alpha1.BackupSchedule{})
+
+ return err
+}
+
+// DeleteCollection deletes a collection of objects.
+func (c *FakeBackupSchedules) DeleteCollection(ctx context.Context, opts v1.DeleteOptions, listOpts v1.ListOptions) error {
+ action := testing.NewDeleteCollectionAction(backupschedulesResource, c.ns, listOpts)
+
+ _, err := c.Fake.Invokes(action, &v1alpha1.BackupScheduleList{})
+ return err
+}
+
+// Patch applies the patch and returns the patched backupSchedule.
+func (c *FakeBackupSchedules) Patch(ctx context.Context, name string, pt types.PatchType, data []byte, opts v1.PatchOptions, subresources ...string) (result *v1alpha1.BackupSchedule, err error) {
+ obj, err := c.Fake.
+ Invokes(testing.NewPatchSubresourceAction(backupschedulesResource, c.ns, name, pt, data, subresources...), &v1alpha1.BackupSchedule{})
+
+ if obj == nil {
+ return nil, err
+ }
+ return obj.(*v1alpha1.BackupSchedule), err
+}
diff --git a/pkg/client/clientset/versioned/typed/dataprotection/v1alpha1/fake/fake_backuptool.go b/pkg/client/clientset/versioned/typed/dataprotection/v1alpha1/fake/fake_backuptool.go
deleted file mode 100644
index 668f8def63d..00000000000
--- a/pkg/client/clientset/versioned/typed/dataprotection/v1alpha1/fake/fake_backuptool.go
+++ /dev/null
@@ -1,133 +0,0 @@
-/*
-Copyright (C) 2022-2023 ApeCloud Co., Ltd
-
-Licensed under the Apache License, Version 2.0 (the "License");
-you may not use this file except in compliance with the License.
-You may obtain a copy of the License at
-
- http://www.apache.org/licenses/LICENSE-2.0
-
-Unless required by applicable law or agreed to in writing, software
-distributed under the License is distributed on an "AS IS" BASIS,
-WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
-See the License for the specific language governing permissions and
-limitations under the License.
-*/
-
-// Code generated by client-gen. DO NOT EDIT.
-
-package fake
-
-import (
- "context"
-
- v1alpha1 "github.com/apecloud/kubeblocks/apis/dataprotection/v1alpha1"
- v1 "k8s.io/apimachinery/pkg/apis/meta/v1"
- labels "k8s.io/apimachinery/pkg/labels"
- schema "k8s.io/apimachinery/pkg/runtime/schema"
- types "k8s.io/apimachinery/pkg/types"
- watch "k8s.io/apimachinery/pkg/watch"
- testing "k8s.io/client-go/testing"
-)
-
-// FakeBackupTools implements BackupToolInterface
-type FakeBackupTools struct {
- Fake *FakeDataprotectionV1alpha1
-}
-
-var backuptoolsResource = schema.GroupVersionResource{Group: "dataprotection.kubeblocks.io", Version: "v1alpha1", Resource: "backuptools"}
-
-var backuptoolsKind = schema.GroupVersionKind{Group: "dataprotection.kubeblocks.io", Version: "v1alpha1", Kind: "BackupTool"}
-
-// Get takes name of the backupTool, and returns the corresponding backupTool object, and an error if there is any.
-func (c *FakeBackupTools) Get(ctx context.Context, name string, options v1.GetOptions) (result *v1alpha1.BackupTool, err error) {
- obj, err := c.Fake.
- Invokes(testing.NewRootGetAction(backuptoolsResource, name), &v1alpha1.BackupTool{})
- if obj == nil {
- return nil, err
- }
- return obj.(*v1alpha1.BackupTool), err
-}
-
-// List takes label and field selectors, and returns the list of BackupTools that match those selectors.
-func (c *FakeBackupTools) List(ctx context.Context, opts v1.ListOptions) (result *v1alpha1.BackupToolList, err error) {
- obj, err := c.Fake.
- Invokes(testing.NewRootListAction(backuptoolsResource, backuptoolsKind, opts), &v1alpha1.BackupToolList{})
- if obj == nil {
- return nil, err
- }
-
- label, _, _ := testing.ExtractFromListOptions(opts)
- if label == nil {
- label = labels.Everything()
- }
- list := &v1alpha1.BackupToolList{ListMeta: obj.(*v1alpha1.BackupToolList).ListMeta}
- for _, item := range obj.(*v1alpha1.BackupToolList).Items {
- if label.Matches(labels.Set(item.Labels)) {
- list.Items = append(list.Items, item)
- }
- }
- return list, err
-}
-
-// Watch returns a watch.Interface that watches the requested backupTools.
-func (c *FakeBackupTools) Watch(ctx context.Context, opts v1.ListOptions) (watch.Interface, error) {
- return c.Fake.
- InvokesWatch(testing.NewRootWatchAction(backuptoolsResource, opts))
-}
-
-// Create takes the representation of a backupTool and creates it. Returns the server's representation of the backupTool, and an error, if there is any.
-func (c *FakeBackupTools) Create(ctx context.Context, backupTool *v1alpha1.BackupTool, opts v1.CreateOptions) (result *v1alpha1.BackupTool, err error) {
- obj, err := c.Fake.
- Invokes(testing.NewRootCreateAction(backuptoolsResource, backupTool), &v1alpha1.BackupTool{})
- if obj == nil {
- return nil, err
- }
- return obj.(*v1alpha1.BackupTool), err
-}
-
-// Update takes the representation of a backupTool and updates it. Returns the server's representation of the backupTool, and an error, if there is any.
-func (c *FakeBackupTools) Update(ctx context.Context, backupTool *v1alpha1.BackupTool, opts v1.UpdateOptions) (result *v1alpha1.BackupTool, err error) {
- obj, err := c.Fake.
- Invokes(testing.NewRootUpdateAction(backuptoolsResource, backupTool), &v1alpha1.BackupTool{})
- if obj == nil {
- return nil, err
- }
- return obj.(*v1alpha1.BackupTool), err
-}
-
-// UpdateStatus was generated because the type contains a Status member.
-// Add a +genclient:noStatus comment above the type to avoid generating UpdateStatus().
-func (c *FakeBackupTools) UpdateStatus(ctx context.Context, backupTool *v1alpha1.BackupTool, opts v1.UpdateOptions) (*v1alpha1.BackupTool, error) {
- obj, err := c.Fake.
- Invokes(testing.NewRootUpdateSubresourceAction(backuptoolsResource, "status", backupTool), &v1alpha1.BackupTool{})
- if obj == nil {
- return nil, err
- }
- return obj.(*v1alpha1.BackupTool), err
-}
-
-// Delete takes name of the backupTool and deletes it. Returns an error if one occurs.
-func (c *FakeBackupTools) Delete(ctx context.Context, name string, opts v1.DeleteOptions) error {
- _, err := c.Fake.
- Invokes(testing.NewRootDeleteActionWithOptions(backuptoolsResource, name, opts), &v1alpha1.BackupTool{})
- return err
-}
-
-// DeleteCollection deletes a collection of objects.
-func (c *FakeBackupTools) DeleteCollection(ctx context.Context, opts v1.DeleteOptions, listOpts v1.ListOptions) error {
- action := testing.NewRootDeleteCollectionAction(backuptoolsResource, listOpts)
-
- _, err := c.Fake.Invokes(action, &v1alpha1.BackupToolList{})
- return err
-}
-
-// Patch applies the patch and returns the patched backupTool.
-func (c *FakeBackupTools) Patch(ctx context.Context, name string, pt types.PatchType, data []byte, opts v1.PatchOptions, subresources ...string) (result *v1alpha1.BackupTool, err error) {
- obj, err := c.Fake.
- Invokes(testing.NewRootPatchSubresourceAction(backuptoolsResource, name, pt, data, subresources...), &v1alpha1.BackupTool{})
- if obj == nil {
- return nil, err
- }
- return obj.(*v1alpha1.BackupTool), err
-}
diff --git a/pkg/client/clientset/versioned/typed/dataprotection/v1alpha1/fake/fake_dataprotection_client.go b/pkg/client/clientset/versioned/typed/dataprotection/v1alpha1/fake/fake_dataprotection_client.go
index f67c522d124..c13966489f8 100644
--- a/pkg/client/clientset/versioned/typed/dataprotection/v1alpha1/fake/fake_dataprotection_client.go
+++ b/pkg/client/clientset/versioned/typed/dataprotection/v1alpha1/fake/fake_dataprotection_client.go
@@ -28,6 +28,10 @@ type FakeDataprotectionV1alpha1 struct {
*testing.Fake
}
+func (c *FakeDataprotectionV1alpha1) ActionSets() v1alpha1.ActionSetInterface {
+ return &FakeActionSets{c}
+}
+
func (c *FakeDataprotectionV1alpha1) Backups(namespace string) v1alpha1.BackupInterface {
return &FakeBackups{c, namespace}
}
@@ -40,12 +44,12 @@ func (c *FakeDataprotectionV1alpha1) BackupRepos() v1alpha1.BackupRepoInterface
return &FakeBackupRepos{c}
}
-func (c *FakeDataprotectionV1alpha1) BackupTools() v1alpha1.BackupToolInterface {
- return &FakeBackupTools{c}
+func (c *FakeDataprotectionV1alpha1) BackupSchedules(namespace string) v1alpha1.BackupScheduleInterface {
+ return &FakeBackupSchedules{c, namespace}
}
-func (c *FakeDataprotectionV1alpha1) RestoreJobs(namespace string) v1alpha1.RestoreJobInterface {
- return &FakeRestoreJobs{c, namespace}
+func (c *FakeDataprotectionV1alpha1) Restores(namespace string) v1alpha1.RestoreInterface {
+ return &FakeRestores{c, namespace}
}
// RESTClient returns a RESTClient that is used to communicate
diff --git a/pkg/client/clientset/versioned/typed/dataprotection/v1alpha1/fake/fake_restore.go b/pkg/client/clientset/versioned/typed/dataprotection/v1alpha1/fake/fake_restore.go
new file mode 100644
index 00000000000..9925bdea1ba
--- /dev/null
+++ b/pkg/client/clientset/versioned/typed/dataprotection/v1alpha1/fake/fake_restore.go
@@ -0,0 +1,141 @@
+/*
+Copyright (C) 2022-2023 ApeCloud Co., Ltd
+
+Licensed under the Apache License, Version 2.0 (the "License");
+you may not use this file except in compliance with the License.
+You may obtain a copy of the License at
+
+ http://www.apache.org/licenses/LICENSE-2.0
+
+Unless required by applicable law or agreed to in writing, software
+distributed under the License is distributed on an "AS IS" BASIS,
+WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+See the License for the specific language governing permissions and
+limitations under the License.
+*/
+
+// Code generated by client-gen. DO NOT EDIT.
+
+package fake
+
+import (
+ "context"
+
+ v1alpha1 "github.com/apecloud/kubeblocks/apis/dataprotection/v1alpha1"
+ v1 "k8s.io/apimachinery/pkg/apis/meta/v1"
+ labels "k8s.io/apimachinery/pkg/labels"
+ types "k8s.io/apimachinery/pkg/types"
+ watch "k8s.io/apimachinery/pkg/watch"
+ testing "k8s.io/client-go/testing"
+)
+
+// FakeRestores implements RestoreInterface
+type FakeRestores struct {
+ Fake *FakeDataprotectionV1alpha1
+ ns string
+}
+
+var restoresResource = v1alpha1.SchemeGroupVersion.WithResource("restores")
+
+var restoresKind = v1alpha1.SchemeGroupVersion.WithKind("Restore")
+
+// Get takes name of the restore, and returns the corresponding restore object, and an error if there is any.
+func (c *FakeRestores) Get(ctx context.Context, name string, options v1.GetOptions) (result *v1alpha1.Restore, err error) {
+ obj, err := c.Fake.
+ Invokes(testing.NewGetAction(restoresResource, c.ns, name), &v1alpha1.Restore{})
+
+ if obj == nil {
+ return nil, err
+ }
+ return obj.(*v1alpha1.Restore), err
+}
+
+// List takes label and field selectors, and returns the list of Restores that match those selectors.
+func (c *FakeRestores) List(ctx context.Context, opts v1.ListOptions) (result *v1alpha1.RestoreList, err error) {
+ obj, err := c.Fake.
+ Invokes(testing.NewListAction(restoresResource, restoresKind, c.ns, opts), &v1alpha1.RestoreList{})
+
+ if obj == nil {
+ return nil, err
+ }
+
+ label, _, _ := testing.ExtractFromListOptions(opts)
+ if label == nil {
+ label = labels.Everything()
+ }
+ list := &v1alpha1.RestoreList{ListMeta: obj.(*v1alpha1.RestoreList).ListMeta}
+ for _, item := range obj.(*v1alpha1.RestoreList).Items {
+ if label.Matches(labels.Set(item.Labels)) {
+ list.Items = append(list.Items, item)
+ }
+ }
+ return list, err
+}
+
+// Watch returns a watch.Interface that watches the requested restores.
+func (c *FakeRestores) Watch(ctx context.Context, opts v1.ListOptions) (watch.Interface, error) {
+ return c.Fake.
+ InvokesWatch(testing.NewWatchAction(restoresResource, c.ns, opts))
+
+}
+
+// Create takes the representation of a restore and creates it. Returns the server's representation of the restore, and an error, if there is any.
+func (c *FakeRestores) Create(ctx context.Context, restore *v1alpha1.Restore, opts v1.CreateOptions) (result *v1alpha1.Restore, err error) {
+ obj, err := c.Fake.
+ Invokes(testing.NewCreateAction(restoresResource, c.ns, restore), &v1alpha1.Restore{})
+
+ if obj == nil {
+ return nil, err
+ }
+ return obj.(*v1alpha1.Restore), err
+}
+
+// Update takes the representation of a restore and updates it. Returns the server's representation of the restore, and an error, if there is any.
+func (c *FakeRestores) Update(ctx context.Context, restore *v1alpha1.Restore, opts v1.UpdateOptions) (result *v1alpha1.Restore, err error) {
+ obj, err := c.Fake.
+ Invokes(testing.NewUpdateAction(restoresResource, c.ns, restore), &v1alpha1.Restore{})
+
+ if obj == nil {
+ return nil, err
+ }
+ return obj.(*v1alpha1.Restore), err
+}
+
+// UpdateStatus was generated because the type contains a Status member.
+// Add a +genclient:noStatus comment above the type to avoid generating UpdateStatus().
+func (c *FakeRestores) UpdateStatus(ctx context.Context, restore *v1alpha1.Restore, opts v1.UpdateOptions) (*v1alpha1.Restore, error) {
+ obj, err := c.Fake.
+ Invokes(testing.NewUpdateSubresourceAction(restoresResource, "status", c.ns, restore), &v1alpha1.Restore{})
+
+ if obj == nil {
+ return nil, err
+ }
+ return obj.(*v1alpha1.Restore), err
+}
+
+// Delete takes name of the restore and deletes it. Returns an error if one occurs.
+func (c *FakeRestores) Delete(ctx context.Context, name string, opts v1.DeleteOptions) error {
+ _, err := c.Fake.
+ Invokes(testing.NewDeleteActionWithOptions(restoresResource, c.ns, name, opts), &v1alpha1.Restore{})
+
+ return err
+}
+
+// DeleteCollection deletes a collection of objects.
+func (c *FakeRestores) DeleteCollection(ctx context.Context, opts v1.DeleteOptions, listOpts v1.ListOptions) error {
+ action := testing.NewDeleteCollectionAction(restoresResource, c.ns, listOpts)
+
+ _, err := c.Fake.Invokes(action, &v1alpha1.RestoreList{})
+ return err
+}
+
+// Patch applies the patch and returns the patched restore.
+func (c *FakeRestores) Patch(ctx context.Context, name string, pt types.PatchType, data []byte, opts v1.PatchOptions, subresources ...string) (result *v1alpha1.Restore, err error) {
+ obj, err := c.Fake.
+ Invokes(testing.NewPatchSubresourceAction(restoresResource, c.ns, name, pt, data, subresources...), &v1alpha1.Restore{})
+
+ if obj == nil {
+ return nil, err
+ }
+ return obj.(*v1alpha1.Restore), err
+}
diff --git a/pkg/client/clientset/versioned/typed/dataprotection/v1alpha1/fake/fake_restorejob.go b/pkg/client/clientset/versioned/typed/dataprotection/v1alpha1/fake/fake_restorejob.go
deleted file mode 100644
index 31d60fbde83..00000000000
--- a/pkg/client/clientset/versioned/typed/dataprotection/v1alpha1/fake/fake_restorejob.go
+++ /dev/null
@@ -1,142 +0,0 @@
-/*
-Copyright (C) 2022-2023 ApeCloud Co., Ltd
-
-Licensed under the Apache License, Version 2.0 (the "License");
-you may not use this file except in compliance with the License.
-You may obtain a copy of the License at
-
- http://www.apache.org/licenses/LICENSE-2.0
-
-Unless required by applicable law or agreed to in writing, software
-distributed under the License is distributed on an "AS IS" BASIS,
-WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
-See the License for the specific language governing permissions and
-limitations under the License.
-*/
-
-// Code generated by client-gen. DO NOT EDIT.
-
-package fake
-
-import (
- "context"
-
- v1alpha1 "github.com/apecloud/kubeblocks/apis/dataprotection/v1alpha1"
- v1 "k8s.io/apimachinery/pkg/apis/meta/v1"
- labels "k8s.io/apimachinery/pkg/labels"
- schema "k8s.io/apimachinery/pkg/runtime/schema"
- types "k8s.io/apimachinery/pkg/types"
- watch "k8s.io/apimachinery/pkg/watch"
- testing "k8s.io/client-go/testing"
-)
-
-// FakeRestoreJobs implements RestoreJobInterface
-type FakeRestoreJobs struct {
- Fake *FakeDataprotectionV1alpha1
- ns string
-}
-
-var restorejobsResource = schema.GroupVersionResource{Group: "dataprotection.kubeblocks.io", Version: "v1alpha1", Resource: "restorejobs"}
-
-var restorejobsKind = schema.GroupVersionKind{Group: "dataprotection.kubeblocks.io", Version: "v1alpha1", Kind: "RestoreJob"}
-
-// Get takes name of the restoreJob, and returns the corresponding restoreJob object, and an error if there is any.
-func (c *FakeRestoreJobs) Get(ctx context.Context, name string, options v1.GetOptions) (result *v1alpha1.RestoreJob, err error) {
- obj, err := c.Fake.
- Invokes(testing.NewGetAction(restorejobsResource, c.ns, name), &v1alpha1.RestoreJob{})
-
- if obj == nil {
- return nil, err
- }
- return obj.(*v1alpha1.RestoreJob), err
-}
-
-// List takes label and field selectors, and returns the list of RestoreJobs that match those selectors.
-func (c *FakeRestoreJobs) List(ctx context.Context, opts v1.ListOptions) (result *v1alpha1.RestoreJobList, err error) {
- obj, err := c.Fake.
- Invokes(testing.NewListAction(restorejobsResource, restorejobsKind, c.ns, opts), &v1alpha1.RestoreJobList{})
-
- if obj == nil {
- return nil, err
- }
-
- label, _, _ := testing.ExtractFromListOptions(opts)
- if label == nil {
- label = labels.Everything()
- }
- list := &v1alpha1.RestoreJobList{ListMeta: obj.(*v1alpha1.RestoreJobList).ListMeta}
- for _, item := range obj.(*v1alpha1.RestoreJobList).Items {
- if label.Matches(labels.Set(item.Labels)) {
- list.Items = append(list.Items, item)
- }
- }
- return list, err
-}
-
-// Watch returns a watch.Interface that watches the requested restoreJobs.
-func (c *FakeRestoreJobs) Watch(ctx context.Context, opts v1.ListOptions) (watch.Interface, error) {
- return c.Fake.
- InvokesWatch(testing.NewWatchAction(restorejobsResource, c.ns, opts))
-
-}
-
-// Create takes the representation of a restoreJob and creates it. Returns the server's representation of the restoreJob, and an error, if there is any.
-func (c *FakeRestoreJobs) Create(ctx context.Context, restoreJob *v1alpha1.RestoreJob, opts v1.CreateOptions) (result *v1alpha1.RestoreJob, err error) {
- obj, err := c.Fake.
- Invokes(testing.NewCreateAction(restorejobsResource, c.ns, restoreJob), &v1alpha1.RestoreJob{})
-
- if obj == nil {
- return nil, err
- }
- return obj.(*v1alpha1.RestoreJob), err
-}
-
-// Update takes the representation of a restoreJob and updates it. Returns the server's representation of the restoreJob, and an error, if there is any.
-func (c *FakeRestoreJobs) Update(ctx context.Context, restoreJob *v1alpha1.RestoreJob, opts v1.UpdateOptions) (result *v1alpha1.RestoreJob, err error) {
- obj, err := c.Fake.
- Invokes(testing.NewUpdateAction(restorejobsResource, c.ns, restoreJob), &v1alpha1.RestoreJob{})
-
- if obj == nil {
- return nil, err
- }
- return obj.(*v1alpha1.RestoreJob), err
-}
-
-// UpdateStatus was generated because the type contains a Status member.
-// Add a +genclient:noStatus comment above the type to avoid generating UpdateStatus().
-func (c *FakeRestoreJobs) UpdateStatus(ctx context.Context, restoreJob *v1alpha1.RestoreJob, opts v1.UpdateOptions) (*v1alpha1.RestoreJob, error) {
- obj, err := c.Fake.
- Invokes(testing.NewUpdateSubresourceAction(restorejobsResource, "status", c.ns, restoreJob), &v1alpha1.RestoreJob{})
-
- if obj == nil {
- return nil, err
- }
- return obj.(*v1alpha1.RestoreJob), err
-}
-
-// Delete takes name of the restoreJob and deletes it. Returns an error if one occurs.
-func (c *FakeRestoreJobs) Delete(ctx context.Context, name string, opts v1.DeleteOptions) error {
- _, err := c.Fake.
- Invokes(testing.NewDeleteActionWithOptions(restorejobsResource, c.ns, name, opts), &v1alpha1.RestoreJob{})
-
- return err
-}
-
-// DeleteCollection deletes a collection of objects.
-func (c *FakeRestoreJobs) DeleteCollection(ctx context.Context, opts v1.DeleteOptions, listOpts v1.ListOptions) error {
- action := testing.NewDeleteCollectionAction(restorejobsResource, c.ns, listOpts)
-
- _, err := c.Fake.Invokes(action, &v1alpha1.RestoreJobList{})
- return err
-}
-
-// Patch applies the patch and returns the patched restoreJob.
-func (c *FakeRestoreJobs) Patch(ctx context.Context, name string, pt types.PatchType, data []byte, opts v1.PatchOptions, subresources ...string) (result *v1alpha1.RestoreJob, err error) {
- obj, err := c.Fake.
- Invokes(testing.NewPatchSubresourceAction(restorejobsResource, c.ns, name, pt, data, subresources...), &v1alpha1.RestoreJob{})
-
- if obj == nil {
- return nil, err
- }
- return obj.(*v1alpha1.RestoreJob), err
-}
diff --git a/pkg/client/clientset/versioned/typed/dataprotection/v1alpha1/generated_expansion.go b/pkg/client/clientset/versioned/typed/dataprotection/v1alpha1/generated_expansion.go
index 6e5cd155273..594323b5e78 100644
--- a/pkg/client/clientset/versioned/typed/dataprotection/v1alpha1/generated_expansion.go
+++ b/pkg/client/clientset/versioned/typed/dataprotection/v1alpha1/generated_expansion.go
@@ -18,12 +18,14 @@ limitations under the License.
package v1alpha1
+type ActionSetExpansion interface{}
+
type BackupExpansion interface{}
type BackupPolicyExpansion interface{}
type BackupRepoExpansion interface{}
-type BackupToolExpansion interface{}
+type BackupScheduleExpansion interface{}
-type RestoreJobExpansion interface{}
+type RestoreExpansion interface{}
diff --git a/pkg/client/clientset/versioned/typed/dataprotection/v1alpha1/restorejob.go b/pkg/client/clientset/versioned/typed/dataprotection/v1alpha1/restore.go
similarity index 52%
rename from pkg/client/clientset/versioned/typed/dataprotection/v1alpha1/restorejob.go
rename to pkg/client/clientset/versioned/typed/dataprotection/v1alpha1/restore.go
index 79455ce5f62..923e5a94115 100644
--- a/pkg/client/clientset/versioned/typed/dataprotection/v1alpha1/restorejob.go
+++ b/pkg/client/clientset/versioned/typed/dataprotection/v1alpha1/restore.go
@@ -30,46 +30,46 @@ import (
rest "k8s.io/client-go/rest"
)
-// RestoreJobsGetter has a method to return a RestoreJobInterface.
+// RestoresGetter has a method to return a RestoreInterface.
// A group's client should implement this interface.
-type RestoreJobsGetter interface {
- RestoreJobs(namespace string) RestoreJobInterface
+type RestoresGetter interface {
+ Restores(namespace string) RestoreInterface
}
-// RestoreJobInterface has methods to work with RestoreJob resources.
-type RestoreJobInterface interface {
- Create(ctx context.Context, restoreJob *v1alpha1.RestoreJob, opts v1.CreateOptions) (*v1alpha1.RestoreJob, error)
- Update(ctx context.Context, restoreJob *v1alpha1.RestoreJob, opts v1.UpdateOptions) (*v1alpha1.RestoreJob, error)
- UpdateStatus(ctx context.Context, restoreJob *v1alpha1.RestoreJob, opts v1.UpdateOptions) (*v1alpha1.RestoreJob, error)
+// RestoreInterface has methods to work with Restore resources.
+type RestoreInterface interface {
+ Create(ctx context.Context, restore *v1alpha1.Restore, opts v1.CreateOptions) (*v1alpha1.Restore, error)
+ Update(ctx context.Context, restore *v1alpha1.Restore, opts v1.UpdateOptions) (*v1alpha1.Restore, error)
+ UpdateStatus(ctx context.Context, restore *v1alpha1.Restore, opts v1.UpdateOptions) (*v1alpha1.Restore, error)
Delete(ctx context.Context, name string, opts v1.DeleteOptions) error
DeleteCollection(ctx context.Context, opts v1.DeleteOptions, listOpts v1.ListOptions) error
- Get(ctx context.Context, name string, opts v1.GetOptions) (*v1alpha1.RestoreJob, error)
- List(ctx context.Context, opts v1.ListOptions) (*v1alpha1.RestoreJobList, error)
+ Get(ctx context.Context, name string, opts v1.GetOptions) (*v1alpha1.Restore, error)
+ List(ctx context.Context, opts v1.ListOptions) (*v1alpha1.RestoreList, error)
Watch(ctx context.Context, opts v1.ListOptions) (watch.Interface, error)
- Patch(ctx context.Context, name string, pt types.PatchType, data []byte, opts v1.PatchOptions, subresources ...string) (result *v1alpha1.RestoreJob, err error)
- RestoreJobExpansion
+ Patch(ctx context.Context, name string, pt types.PatchType, data []byte, opts v1.PatchOptions, subresources ...string) (result *v1alpha1.Restore, err error)
+ RestoreExpansion
}
-// restoreJobs implements RestoreJobInterface
-type restoreJobs struct {
+// restores implements RestoreInterface
+type restores struct {
client rest.Interface
ns string
}
-// newRestoreJobs returns a RestoreJobs
-func newRestoreJobs(c *DataprotectionV1alpha1Client, namespace string) *restoreJobs {
- return &restoreJobs{
+// newRestores returns a Restores
+func newRestores(c *DataprotectionV1alpha1Client, namespace string) *restores {
+ return &restores{
client: c.RESTClient(),
ns: namespace,
}
}
-// Get takes name of the restoreJob, and returns the corresponding restoreJob object, and an error if there is any.
-func (c *restoreJobs) Get(ctx context.Context, name string, options v1.GetOptions) (result *v1alpha1.RestoreJob, err error) {
- result = &v1alpha1.RestoreJob{}
+// Get takes name of the restore, and returns the corresponding restore object, and an error if there is any.
+func (c *restores) Get(ctx context.Context, name string, options v1.GetOptions) (result *v1alpha1.Restore, err error) {
+ result = &v1alpha1.Restore{}
err = c.client.Get().
Namespace(c.ns).
- Resource("restorejobs").
+ Resource("restores").
Name(name).
VersionedParams(&options, scheme.ParameterCodec).
Do(ctx).
@@ -77,16 +77,16 @@ func (c *restoreJobs) Get(ctx context.Context, name string, options v1.GetOption
return
}
-// List takes label and field selectors, and returns the list of RestoreJobs that match those selectors.
-func (c *restoreJobs) List(ctx context.Context, opts v1.ListOptions) (result *v1alpha1.RestoreJobList, err error) {
+// List takes label and field selectors, and returns the list of Restores that match those selectors.
+func (c *restores) List(ctx context.Context, opts v1.ListOptions) (result *v1alpha1.RestoreList, err error) {
var timeout time.Duration
if opts.TimeoutSeconds != nil {
timeout = time.Duration(*opts.TimeoutSeconds) * time.Second
}
- result = &v1alpha1.RestoreJobList{}
+ result = &v1alpha1.RestoreList{}
err = c.client.Get().
Namespace(c.ns).
- Resource("restorejobs").
+ Resource("restores").
VersionedParams(&opts, scheme.ParameterCodec).
Timeout(timeout).
Do(ctx).
@@ -94,8 +94,8 @@ func (c *restoreJobs) List(ctx context.Context, opts v1.ListOptions) (result *v1
return
}
-// Watch returns a watch.Interface that watches the requested restoreJobs.
-func (c *restoreJobs) Watch(ctx context.Context, opts v1.ListOptions) (watch.Interface, error) {
+// Watch returns a watch.Interface that watches the requested restores.
+func (c *restores) Watch(ctx context.Context, opts v1.ListOptions) (watch.Interface, error) {
var timeout time.Duration
if opts.TimeoutSeconds != nil {
timeout = time.Duration(*opts.TimeoutSeconds) * time.Second
@@ -103,34 +103,34 @@ func (c *restoreJobs) Watch(ctx context.Context, opts v1.ListOptions) (watch.Int
opts.Watch = true
return c.client.Get().
Namespace(c.ns).
- Resource("restorejobs").
+ Resource("restores").
VersionedParams(&opts, scheme.ParameterCodec).
Timeout(timeout).
Watch(ctx)
}
-// Create takes the representation of a restoreJob and creates it. Returns the server's representation of the restoreJob, and an error, if there is any.
-func (c *restoreJobs) Create(ctx context.Context, restoreJob *v1alpha1.RestoreJob, opts v1.CreateOptions) (result *v1alpha1.RestoreJob, err error) {
- result = &v1alpha1.RestoreJob{}
+// Create takes the representation of a restore and creates it. Returns the server's representation of the restore, and an error, if there is any.
+func (c *restores) Create(ctx context.Context, restore *v1alpha1.Restore, opts v1.CreateOptions) (result *v1alpha1.Restore, err error) {
+ result = &v1alpha1.Restore{}
err = c.client.Post().
Namespace(c.ns).
- Resource("restorejobs").
+ Resource("restores").
VersionedParams(&opts, scheme.ParameterCodec).
- Body(restoreJob).
+ Body(restore).
Do(ctx).
Into(result)
return
}
-// Update takes the representation of a restoreJob and updates it. Returns the server's representation of the restoreJob, and an error, if there is any.
-func (c *restoreJobs) Update(ctx context.Context, restoreJob *v1alpha1.RestoreJob, opts v1.UpdateOptions) (result *v1alpha1.RestoreJob, err error) {
- result = &v1alpha1.RestoreJob{}
+// Update takes the representation of a restore and updates it. Returns the server's representation of the restore, and an error, if there is any.
+func (c *restores) Update(ctx context.Context, restore *v1alpha1.Restore, opts v1.UpdateOptions) (result *v1alpha1.Restore, err error) {
+ result = &v1alpha1.Restore{}
err = c.client.Put().
Namespace(c.ns).
- Resource("restorejobs").
- Name(restoreJob.Name).
+ Resource("restores").
+ Name(restore.Name).
VersionedParams(&opts, scheme.ParameterCodec).
- Body(restoreJob).
+ Body(restore).
Do(ctx).
Into(result)
return
@@ -138,25 +138,25 @@ func (c *restoreJobs) Update(ctx context.Context, restoreJob *v1alpha1.RestoreJo
// UpdateStatus was generated because the type contains a Status member.
// Add a +genclient:noStatus comment above the type to avoid generating UpdateStatus().
-func (c *restoreJobs) UpdateStatus(ctx context.Context, restoreJob *v1alpha1.RestoreJob, opts v1.UpdateOptions) (result *v1alpha1.RestoreJob, err error) {
- result = &v1alpha1.RestoreJob{}
+func (c *restores) UpdateStatus(ctx context.Context, restore *v1alpha1.Restore, opts v1.UpdateOptions) (result *v1alpha1.Restore, err error) {
+ result = &v1alpha1.Restore{}
err = c.client.Put().
Namespace(c.ns).
- Resource("restorejobs").
- Name(restoreJob.Name).
+ Resource("restores").
+ Name(restore.Name).
SubResource("status").
VersionedParams(&opts, scheme.ParameterCodec).
- Body(restoreJob).
+ Body(restore).
Do(ctx).
Into(result)
return
}
-// Delete takes name of the restoreJob and deletes it. Returns an error if one occurs.
-func (c *restoreJobs) Delete(ctx context.Context, name string, opts v1.DeleteOptions) error {
+// Delete takes name of the restore and deletes it. Returns an error if one occurs.
+func (c *restores) Delete(ctx context.Context, name string, opts v1.DeleteOptions) error {
return c.client.Delete().
Namespace(c.ns).
- Resource("restorejobs").
+ Resource("restores").
Name(name).
Body(&opts).
Do(ctx).
@@ -164,14 +164,14 @@ func (c *restoreJobs) Delete(ctx context.Context, name string, opts v1.DeleteOpt
}
// DeleteCollection deletes a collection of objects.
-func (c *restoreJobs) DeleteCollection(ctx context.Context, opts v1.DeleteOptions, listOpts v1.ListOptions) error {
+func (c *restores) DeleteCollection(ctx context.Context, opts v1.DeleteOptions, listOpts v1.ListOptions) error {
var timeout time.Duration
if listOpts.TimeoutSeconds != nil {
timeout = time.Duration(*listOpts.TimeoutSeconds) * time.Second
}
return c.client.Delete().
Namespace(c.ns).
- Resource("restorejobs").
+ Resource("restores").
VersionedParams(&listOpts, scheme.ParameterCodec).
Timeout(timeout).
Body(&opts).
@@ -179,12 +179,12 @@ func (c *restoreJobs) DeleteCollection(ctx context.Context, opts v1.DeleteOption
Error()
}
-// Patch applies the patch and returns the patched restoreJob.
-func (c *restoreJobs) Patch(ctx context.Context, name string, pt types.PatchType, data []byte, opts v1.PatchOptions, subresources ...string) (result *v1alpha1.RestoreJob, err error) {
- result = &v1alpha1.RestoreJob{}
+// Patch applies the patch and returns the patched restore.
+func (c *restores) Patch(ctx context.Context, name string, pt types.PatchType, data []byte, opts v1.PatchOptions, subresources ...string) (result *v1alpha1.Restore, err error) {
+ result = &v1alpha1.Restore{}
err = c.client.Patch(pt).
Namespace(c.ns).
- Resource("restorejobs").
+ Resource("restores").
Name(name).
SubResource(subresources...).
VersionedParams(&opts, scheme.ParameterCodec).
diff --git a/pkg/client/clientset/versioned/typed/extensions/v1alpha1/fake/fake_addon.go b/pkg/client/clientset/versioned/typed/extensions/v1alpha1/fake/fake_addon.go
index 866ad303948..c7368d9fa9f 100644
--- a/pkg/client/clientset/versioned/typed/extensions/v1alpha1/fake/fake_addon.go
+++ b/pkg/client/clientset/versioned/typed/extensions/v1alpha1/fake/fake_addon.go
@@ -24,7 +24,6 @@ import (
v1alpha1 "github.com/apecloud/kubeblocks/apis/extensions/v1alpha1"
v1 "k8s.io/apimachinery/pkg/apis/meta/v1"
labels "k8s.io/apimachinery/pkg/labels"
- schema "k8s.io/apimachinery/pkg/runtime/schema"
types "k8s.io/apimachinery/pkg/types"
watch "k8s.io/apimachinery/pkg/watch"
testing "k8s.io/client-go/testing"
@@ -35,9 +34,9 @@ type FakeAddons struct {
Fake *FakeExtensionsV1alpha1
}
-var addonsResource = schema.GroupVersionResource{Group: "extensions.kubeblocks.io", Version: "v1alpha1", Resource: "addons"}
+var addonsResource = v1alpha1.SchemeGroupVersion.WithResource("addons")
-var addonsKind = schema.GroupVersionKind{Group: "extensions.kubeblocks.io", Version: "v1alpha1", Kind: "Addon"}
+var addonsKind = v1alpha1.SchemeGroupVersion.WithKind("Addon")
// Get takes name of the addon, and returns the corresponding addon object, and an error if there is any.
func (c *FakeAddons) Get(ctx context.Context, name string, options v1.GetOptions) (result *v1alpha1.Addon, err error) {
diff --git a/pkg/client/clientset/versioned/typed/storage/v1alpha1/fake/fake_storageprovider.go b/pkg/client/clientset/versioned/typed/storage/v1alpha1/fake/fake_storageprovider.go
index 325a724d114..fb2d05dbee0 100644
--- a/pkg/client/clientset/versioned/typed/storage/v1alpha1/fake/fake_storageprovider.go
+++ b/pkg/client/clientset/versioned/typed/storage/v1alpha1/fake/fake_storageprovider.go
@@ -24,7 +24,6 @@ import (
v1alpha1 "github.com/apecloud/kubeblocks/apis/storage/v1alpha1"
v1 "k8s.io/apimachinery/pkg/apis/meta/v1"
labels "k8s.io/apimachinery/pkg/labels"
- schema "k8s.io/apimachinery/pkg/runtime/schema"
types "k8s.io/apimachinery/pkg/types"
watch "k8s.io/apimachinery/pkg/watch"
testing "k8s.io/client-go/testing"
@@ -35,9 +34,9 @@ type FakeStorageProviders struct {
Fake *FakeStorageV1alpha1
}
-var storageprovidersResource = schema.GroupVersionResource{Group: "storage.kubeblocks.io", Version: "v1alpha1", Resource: "storageproviders"}
+var storageprovidersResource = v1alpha1.SchemeGroupVersion.WithResource("storageproviders")
-var storageprovidersKind = schema.GroupVersionKind{Group: "storage.kubeblocks.io", Version: "v1alpha1", Kind: "StorageProvider"}
+var storageprovidersKind = v1alpha1.SchemeGroupVersion.WithKind("StorageProvider")
// Get takes name of the storageProvider, and returns the corresponding storageProvider object, and an error if there is any.
func (c *FakeStorageProviders) Get(ctx context.Context, name string, options v1.GetOptions) (result *v1alpha1.StorageProvider, err error) {
diff --git a/pkg/client/clientset/versioned/typed/workloads/v1alpha1/fake/fake_replicatedstatemachine.go b/pkg/client/clientset/versioned/typed/workloads/v1alpha1/fake/fake_replicatedstatemachine.go
index 69b962e7026..fb5fded7d03 100644
--- a/pkg/client/clientset/versioned/typed/workloads/v1alpha1/fake/fake_replicatedstatemachine.go
+++ b/pkg/client/clientset/versioned/typed/workloads/v1alpha1/fake/fake_replicatedstatemachine.go
@@ -24,7 +24,6 @@ import (
v1alpha1 "github.com/apecloud/kubeblocks/apis/workloads/v1alpha1"
v1 "k8s.io/apimachinery/pkg/apis/meta/v1"
labels "k8s.io/apimachinery/pkg/labels"
- schema "k8s.io/apimachinery/pkg/runtime/schema"
types "k8s.io/apimachinery/pkg/types"
watch "k8s.io/apimachinery/pkg/watch"
testing "k8s.io/client-go/testing"
@@ -36,9 +35,9 @@ type FakeReplicatedStateMachines struct {
ns string
}
-var replicatedstatemachinesResource = schema.GroupVersionResource{Group: "workloads", Version: "v1alpha1", Resource: "replicatedstatemachines"}
+var replicatedstatemachinesResource = v1alpha1.SchemeGroupVersion.WithResource("replicatedstatemachines")
-var replicatedstatemachinesKind = schema.GroupVersionKind{Group: "workloads", Version: "v1alpha1", Kind: "ReplicatedStateMachine"}
+var replicatedstatemachinesKind = v1alpha1.SchemeGroupVersion.WithKind("ReplicatedStateMachine")
// Get takes name of the replicatedStateMachine, and returns the corresponding replicatedStateMachine object, and an error if there is any.
func (c *FakeReplicatedStateMachines) Get(ctx context.Context, name string, options v1.GetOptions) (result *v1alpha1.ReplicatedStateMachine, err error) {
diff --git a/pkg/client/informers/externalversions/dataprotection/v1alpha1/backuptool.go b/pkg/client/informers/externalversions/dataprotection/v1alpha1/actionset.go
similarity index 58%
rename from pkg/client/informers/externalversions/dataprotection/v1alpha1/backuptool.go
rename to pkg/client/informers/externalversions/dataprotection/v1alpha1/actionset.go
index 4a29ff221f7..edab407a699 100644
--- a/pkg/client/informers/externalversions/dataprotection/v1alpha1/backuptool.go
+++ b/pkg/client/informers/externalversions/dataprotection/v1alpha1/actionset.go
@@ -32,58 +32,58 @@ import (
cache "k8s.io/client-go/tools/cache"
)
-// BackupToolInformer provides access to a shared informer and lister for
-// BackupTools.
-type BackupToolInformer interface {
+// ActionSetInformer provides access to a shared informer and lister for
+// ActionSets.
+type ActionSetInformer interface {
Informer() cache.SharedIndexInformer
- Lister() v1alpha1.BackupToolLister
+ Lister() v1alpha1.ActionSetLister
}
-type backupToolInformer struct {
+type actionSetInformer struct {
factory internalinterfaces.SharedInformerFactory
tweakListOptions internalinterfaces.TweakListOptionsFunc
}
-// NewBackupToolInformer constructs a new informer for BackupTool type.
+// NewActionSetInformer constructs a new informer for ActionSet type.
// Always prefer using an informer factory to get a shared informer instead of getting an independent
// one. This reduces memory footprint and number of connections to the server.
-func NewBackupToolInformer(client versioned.Interface, resyncPeriod time.Duration, indexers cache.Indexers) cache.SharedIndexInformer {
- return NewFilteredBackupToolInformer(client, resyncPeriod, indexers, nil)
+func NewActionSetInformer(client versioned.Interface, resyncPeriod time.Duration, indexers cache.Indexers) cache.SharedIndexInformer {
+ return NewFilteredActionSetInformer(client, resyncPeriod, indexers, nil)
}
-// NewFilteredBackupToolInformer constructs a new informer for BackupTool type.
+// NewFilteredActionSetInformer constructs a new informer for ActionSet type.
// Always prefer using an informer factory to get a shared informer instead of getting an independent
// one. This reduces memory footprint and number of connections to the server.
-func NewFilteredBackupToolInformer(client versioned.Interface, resyncPeriod time.Duration, indexers cache.Indexers, tweakListOptions internalinterfaces.TweakListOptionsFunc) cache.SharedIndexInformer {
+func NewFilteredActionSetInformer(client versioned.Interface, resyncPeriod time.Duration, indexers cache.Indexers, tweakListOptions internalinterfaces.TweakListOptionsFunc) cache.SharedIndexInformer {
return cache.NewSharedIndexInformer(
&cache.ListWatch{
ListFunc: func(options v1.ListOptions) (runtime.Object, error) {
if tweakListOptions != nil {
tweakListOptions(&options)
}
- return client.DataprotectionV1alpha1().BackupTools().List(context.TODO(), options)
+ return client.DataprotectionV1alpha1().ActionSets().List(context.TODO(), options)
},
WatchFunc: func(options v1.ListOptions) (watch.Interface, error) {
if tweakListOptions != nil {
tweakListOptions(&options)
}
- return client.DataprotectionV1alpha1().BackupTools().Watch(context.TODO(), options)
+ return client.DataprotectionV1alpha1().ActionSets().Watch(context.TODO(), options)
},
},
- &dataprotectionv1alpha1.BackupTool{},
+ &dataprotectionv1alpha1.ActionSet{},
resyncPeriod,
indexers,
)
}
-func (f *backupToolInformer) defaultInformer(client versioned.Interface, resyncPeriod time.Duration) cache.SharedIndexInformer {
- return NewFilteredBackupToolInformer(client, resyncPeriod, cache.Indexers{cache.NamespaceIndex: cache.MetaNamespaceIndexFunc}, f.tweakListOptions)
+func (f *actionSetInformer) defaultInformer(client versioned.Interface, resyncPeriod time.Duration) cache.SharedIndexInformer {
+ return NewFilteredActionSetInformer(client, resyncPeriod, cache.Indexers{cache.NamespaceIndex: cache.MetaNamespaceIndexFunc}, f.tweakListOptions)
}
-func (f *backupToolInformer) Informer() cache.SharedIndexInformer {
- return f.factory.InformerFor(&dataprotectionv1alpha1.BackupTool{}, f.defaultInformer)
+func (f *actionSetInformer) Informer() cache.SharedIndexInformer {
+ return f.factory.InformerFor(&dataprotectionv1alpha1.ActionSet{}, f.defaultInformer)
}
-func (f *backupToolInformer) Lister() v1alpha1.BackupToolLister {
- return v1alpha1.NewBackupToolLister(f.Informer().GetIndexer())
+func (f *actionSetInformer) Lister() v1alpha1.ActionSetLister {
+ return v1alpha1.NewActionSetLister(f.Informer().GetIndexer())
}
diff --git a/pkg/client/informers/externalversions/dataprotection/v1alpha1/backupschedule.go b/pkg/client/informers/externalversions/dataprotection/v1alpha1/backupschedule.go
new file mode 100644
index 00000000000..492f03c4fc3
--- /dev/null
+++ b/pkg/client/informers/externalversions/dataprotection/v1alpha1/backupschedule.go
@@ -0,0 +1,90 @@
+/*
+Copyright (C) 2022-2023 ApeCloud Co., Ltd
+
+Licensed under the Apache License, Version 2.0 (the "License");
+you may not use this file except in compliance with the License.
+You may obtain a copy of the License at
+
+ http://www.apache.org/licenses/LICENSE-2.0
+
+Unless required by applicable law or agreed to in writing, software
+distributed under the License is distributed on an "AS IS" BASIS,
+WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+See the License for the specific language governing permissions and
+limitations under the License.
+*/
+
+// Code generated by informer-gen. DO NOT EDIT.
+
+package v1alpha1
+
+import (
+ "context"
+ time "time"
+
+ dataprotectionv1alpha1 "github.com/apecloud/kubeblocks/apis/dataprotection/v1alpha1"
+ versioned "github.com/apecloud/kubeblocks/pkg/client/clientset/versioned"
+ internalinterfaces "github.com/apecloud/kubeblocks/pkg/client/informers/externalversions/internalinterfaces"
+ v1alpha1 "github.com/apecloud/kubeblocks/pkg/client/listers/dataprotection/v1alpha1"
+ v1 "k8s.io/apimachinery/pkg/apis/meta/v1"
+ runtime "k8s.io/apimachinery/pkg/runtime"
+ watch "k8s.io/apimachinery/pkg/watch"
+ cache "k8s.io/client-go/tools/cache"
+)
+
+// BackupScheduleInformer provides access to a shared informer and lister for
+// BackupSchedules.
+type BackupScheduleInformer interface {
+ Informer() cache.SharedIndexInformer
+ Lister() v1alpha1.BackupScheduleLister
+}
+
+type backupScheduleInformer struct {
+ factory internalinterfaces.SharedInformerFactory
+ tweakListOptions internalinterfaces.TweakListOptionsFunc
+ namespace string
+}
+
+// NewBackupScheduleInformer constructs a new informer for BackupSchedule type.
+// Always prefer using an informer factory to get a shared informer instead of getting an independent
+// one. This reduces memory footprint and number of connections to the server.
+func NewBackupScheduleInformer(client versioned.Interface, namespace string, resyncPeriod time.Duration, indexers cache.Indexers) cache.SharedIndexInformer {
+ return NewFilteredBackupScheduleInformer(client, namespace, resyncPeriod, indexers, nil)
+}
+
+// NewFilteredBackupScheduleInformer constructs a new informer for BackupSchedule type.
+// Always prefer using an informer factory to get a shared informer instead of getting an independent
+// one. This reduces memory footprint and number of connections to the server.
+func NewFilteredBackupScheduleInformer(client versioned.Interface, namespace string, resyncPeriod time.Duration, indexers cache.Indexers, tweakListOptions internalinterfaces.TweakListOptionsFunc) cache.SharedIndexInformer {
+ return cache.NewSharedIndexInformer(
+ &cache.ListWatch{
+ ListFunc: func(options v1.ListOptions) (runtime.Object, error) {
+ if tweakListOptions != nil {
+ tweakListOptions(&options)
+ }
+ return client.DataprotectionV1alpha1().BackupSchedules(namespace).List(context.TODO(), options)
+ },
+ WatchFunc: func(options v1.ListOptions) (watch.Interface, error) {
+ if tweakListOptions != nil {
+ tweakListOptions(&options)
+ }
+ return client.DataprotectionV1alpha1().BackupSchedules(namespace).Watch(context.TODO(), options)
+ },
+ },
+ &dataprotectionv1alpha1.BackupSchedule{},
+ resyncPeriod,
+ indexers,
+ )
+}
+
+func (f *backupScheduleInformer) defaultInformer(client versioned.Interface, resyncPeriod time.Duration) cache.SharedIndexInformer {
+ return NewFilteredBackupScheduleInformer(client, f.namespace, resyncPeriod, cache.Indexers{cache.NamespaceIndex: cache.MetaNamespaceIndexFunc}, f.tweakListOptions)
+}
+
+func (f *backupScheduleInformer) Informer() cache.SharedIndexInformer {
+ return f.factory.InformerFor(&dataprotectionv1alpha1.BackupSchedule{}, f.defaultInformer)
+}
+
+func (f *backupScheduleInformer) Lister() v1alpha1.BackupScheduleLister {
+ return v1alpha1.NewBackupScheduleLister(f.Informer().GetIndexer())
+}
diff --git a/pkg/client/informers/externalversions/dataprotection/v1alpha1/interface.go b/pkg/client/informers/externalversions/dataprotection/v1alpha1/interface.go
index 32c098e82d2..54fa5e1423a 100644
--- a/pkg/client/informers/externalversions/dataprotection/v1alpha1/interface.go
+++ b/pkg/client/informers/externalversions/dataprotection/v1alpha1/interface.go
@@ -24,16 +24,18 @@ import (
// Interface provides access to all the informers in this group version.
type Interface interface {
+ // ActionSets returns a ActionSetInformer.
+ ActionSets() ActionSetInformer
// Backups returns a BackupInformer.
Backups() BackupInformer
// BackupPolicies returns a BackupPolicyInformer.
BackupPolicies() BackupPolicyInformer
// BackupRepos returns a BackupRepoInformer.
BackupRepos() BackupRepoInformer
- // BackupTools returns a BackupToolInformer.
- BackupTools() BackupToolInformer
- // RestoreJobs returns a RestoreJobInformer.
- RestoreJobs() RestoreJobInformer
+ // BackupSchedules returns a BackupScheduleInformer.
+ BackupSchedules() BackupScheduleInformer
+ // Restores returns a RestoreInformer.
+ Restores() RestoreInformer
}
type version struct {
@@ -47,6 +49,11 @@ func New(f internalinterfaces.SharedInformerFactory, namespace string, tweakList
return &version{factory: f, namespace: namespace, tweakListOptions: tweakListOptions}
}
+// ActionSets returns a ActionSetInformer.
+func (v *version) ActionSets() ActionSetInformer {
+ return &actionSetInformer{factory: v.factory, tweakListOptions: v.tweakListOptions}
+}
+
// Backups returns a BackupInformer.
func (v *version) Backups() BackupInformer {
return &backupInformer{factory: v.factory, namespace: v.namespace, tweakListOptions: v.tweakListOptions}
@@ -62,12 +69,12 @@ func (v *version) BackupRepos() BackupRepoInformer {
return &backupRepoInformer{factory: v.factory, tweakListOptions: v.tweakListOptions}
}
-// BackupTools returns a BackupToolInformer.
-func (v *version) BackupTools() BackupToolInformer {
- return &backupToolInformer{factory: v.factory, tweakListOptions: v.tweakListOptions}
+// BackupSchedules returns a BackupScheduleInformer.
+func (v *version) BackupSchedules() BackupScheduleInformer {
+ return &backupScheduleInformer{factory: v.factory, namespace: v.namespace, tweakListOptions: v.tweakListOptions}
}
-// RestoreJobs returns a RestoreJobInformer.
-func (v *version) RestoreJobs() RestoreJobInformer {
- return &restoreJobInformer{factory: v.factory, namespace: v.namespace, tweakListOptions: v.tweakListOptions}
+// Restores returns a RestoreInformer.
+func (v *version) Restores() RestoreInformer {
+ return &restoreInformer{factory: v.factory, namespace: v.namespace, tweakListOptions: v.tweakListOptions}
}
diff --git a/pkg/client/informers/externalversions/dataprotection/v1alpha1/restorejob.go b/pkg/client/informers/externalversions/dataprotection/v1alpha1/restore.go
similarity index 57%
rename from pkg/client/informers/externalversions/dataprotection/v1alpha1/restorejob.go
rename to pkg/client/informers/externalversions/dataprotection/v1alpha1/restore.go
index 3a339184313..fe0e68f77c7 100644
--- a/pkg/client/informers/externalversions/dataprotection/v1alpha1/restorejob.go
+++ b/pkg/client/informers/externalversions/dataprotection/v1alpha1/restore.go
@@ -32,59 +32,59 @@ import (
cache "k8s.io/client-go/tools/cache"
)
-// RestoreJobInformer provides access to a shared informer and lister for
-// RestoreJobs.
-type RestoreJobInformer interface {
+// RestoreInformer provides access to a shared informer and lister for
+// Restores.
+type RestoreInformer interface {
Informer() cache.SharedIndexInformer
- Lister() v1alpha1.RestoreJobLister
+ Lister() v1alpha1.RestoreLister
}
-type restoreJobInformer struct {
+type restoreInformer struct {
factory internalinterfaces.SharedInformerFactory
tweakListOptions internalinterfaces.TweakListOptionsFunc
namespace string
}
-// NewRestoreJobInformer constructs a new informer for RestoreJob type.
+// NewRestoreInformer constructs a new informer for Restore type.
// Always prefer using an informer factory to get a shared informer instead of getting an independent
// one. This reduces memory footprint and number of connections to the server.
-func NewRestoreJobInformer(client versioned.Interface, namespace string, resyncPeriod time.Duration, indexers cache.Indexers) cache.SharedIndexInformer {
- return NewFilteredRestoreJobInformer(client, namespace, resyncPeriod, indexers, nil)
+func NewRestoreInformer(client versioned.Interface, namespace string, resyncPeriod time.Duration, indexers cache.Indexers) cache.SharedIndexInformer {
+ return NewFilteredRestoreInformer(client, namespace, resyncPeriod, indexers, nil)
}
-// NewFilteredRestoreJobInformer constructs a new informer for RestoreJob type.
+// NewFilteredRestoreInformer constructs a new informer for Restore type.
// Always prefer using an informer factory to get a shared informer instead of getting an independent
// one. This reduces memory footprint and number of connections to the server.
-func NewFilteredRestoreJobInformer(client versioned.Interface, namespace string, resyncPeriod time.Duration, indexers cache.Indexers, tweakListOptions internalinterfaces.TweakListOptionsFunc) cache.SharedIndexInformer {
+func NewFilteredRestoreInformer(client versioned.Interface, namespace string, resyncPeriod time.Duration, indexers cache.Indexers, tweakListOptions internalinterfaces.TweakListOptionsFunc) cache.SharedIndexInformer {
return cache.NewSharedIndexInformer(
&cache.ListWatch{
ListFunc: func(options v1.ListOptions) (runtime.Object, error) {
if tweakListOptions != nil {
tweakListOptions(&options)
}
- return client.DataprotectionV1alpha1().RestoreJobs(namespace).List(context.TODO(), options)
+ return client.DataprotectionV1alpha1().Restores(namespace).List(context.TODO(), options)
},
WatchFunc: func(options v1.ListOptions) (watch.Interface, error) {
if tweakListOptions != nil {
tweakListOptions(&options)
}
- return client.DataprotectionV1alpha1().RestoreJobs(namespace).Watch(context.TODO(), options)
+ return client.DataprotectionV1alpha1().Restores(namespace).Watch(context.TODO(), options)
},
},
- &dataprotectionv1alpha1.RestoreJob{},
+ &dataprotectionv1alpha1.Restore{},
resyncPeriod,
indexers,
)
}
-func (f *restoreJobInformer) defaultInformer(client versioned.Interface, resyncPeriod time.Duration) cache.SharedIndexInformer {
- return NewFilteredRestoreJobInformer(client, f.namespace, resyncPeriod, cache.Indexers{cache.NamespaceIndex: cache.MetaNamespaceIndexFunc}, f.tweakListOptions)
+func (f *restoreInformer) defaultInformer(client versioned.Interface, resyncPeriod time.Duration) cache.SharedIndexInformer {
+ return NewFilteredRestoreInformer(client, f.namespace, resyncPeriod, cache.Indexers{cache.NamespaceIndex: cache.MetaNamespaceIndexFunc}, f.tweakListOptions)
}
-func (f *restoreJobInformer) Informer() cache.SharedIndexInformer {
- return f.factory.InformerFor(&dataprotectionv1alpha1.RestoreJob{}, f.defaultInformer)
+func (f *restoreInformer) Informer() cache.SharedIndexInformer {
+ return f.factory.InformerFor(&dataprotectionv1alpha1.Restore{}, f.defaultInformer)
}
-func (f *restoreJobInformer) Lister() v1alpha1.RestoreJobLister {
- return v1alpha1.NewRestoreJobLister(f.Informer().GetIndexer())
+func (f *restoreInformer) Lister() v1alpha1.RestoreLister {
+ return v1alpha1.NewRestoreLister(f.Informer().GetIndexer())
}
diff --git a/pkg/client/informers/externalversions/factory.go b/pkg/client/informers/externalversions/factory.go
index 4abf200d5f3..39d453b4acd 100644
--- a/pkg/client/informers/externalversions/factory.go
+++ b/pkg/client/informers/externalversions/factory.go
@@ -170,7 +170,7 @@ func (f *sharedInformerFactory) WaitForCacheSync(stopCh <-chan struct{}) map[ref
return res
}
-// InternalInformerFor returns the SharedIndexInformer for obj using an internal
+// InformerFor returns the SharedIndexInformer for obj using an internal
// client.
func (f *sharedInformerFactory) InformerFor(obj runtime.Object, newFunc internalinterfaces.NewInformerFunc) cache.SharedIndexInformer {
f.lock.Lock()
@@ -243,7 +243,7 @@ type SharedInformerFactory interface {
// ForResource gives generic access to a shared informer of the matching type.
ForResource(resource schema.GroupVersionResource) (GenericInformer, error)
- // InternalInformerFor returns the SharedIndexInformer for obj using an internal
+ // InformerFor returns the SharedIndexInformer for obj using an internal
// client.
InformerFor(obj runtime.Object, newFunc internalinterfaces.NewInformerFunc) cache.SharedIndexInformer
diff --git a/pkg/client/informers/externalversions/generic.go b/pkg/client/informers/externalversions/generic.go
index c18f12b4fc1..3d50c4b4179 100644
--- a/pkg/client/informers/externalversions/generic.go
+++ b/pkg/client/informers/externalversions/generic.go
@@ -77,16 +77,18 @@ func (f *sharedInformerFactory) ForResource(resource schema.GroupVersionResource
return &genericInformer{resource: resource.GroupResource(), informer: f.Apps().V1alpha1().ServiceDescriptors().Informer()}, nil
// Group=dataprotection.kubeblocks.io, Version=v1alpha1
+ case dataprotectionv1alpha1.SchemeGroupVersion.WithResource("actionsets"):
+ return &genericInformer{resource: resource.GroupResource(), informer: f.Dataprotection().V1alpha1().ActionSets().Informer()}, nil
case dataprotectionv1alpha1.SchemeGroupVersion.WithResource("backups"):
return &genericInformer{resource: resource.GroupResource(), informer: f.Dataprotection().V1alpha1().Backups().Informer()}, nil
case dataprotectionv1alpha1.SchemeGroupVersion.WithResource("backuppolicies"):
return &genericInformer{resource: resource.GroupResource(), informer: f.Dataprotection().V1alpha1().BackupPolicies().Informer()}, nil
case dataprotectionv1alpha1.SchemeGroupVersion.WithResource("backuprepos"):
return &genericInformer{resource: resource.GroupResource(), informer: f.Dataprotection().V1alpha1().BackupRepos().Informer()}, nil
- case dataprotectionv1alpha1.SchemeGroupVersion.WithResource("backuptools"):
- return &genericInformer{resource: resource.GroupResource(), informer: f.Dataprotection().V1alpha1().BackupTools().Informer()}, nil
- case dataprotectionv1alpha1.SchemeGroupVersion.WithResource("restorejobs"):
- return &genericInformer{resource: resource.GroupResource(), informer: f.Dataprotection().V1alpha1().RestoreJobs().Informer()}, nil
+ case dataprotectionv1alpha1.SchemeGroupVersion.WithResource("backupschedules"):
+ return &genericInformer{resource: resource.GroupResource(), informer: f.Dataprotection().V1alpha1().BackupSchedules().Informer()}, nil
+ case dataprotectionv1alpha1.SchemeGroupVersion.WithResource("restores"):
+ return &genericInformer{resource: resource.GroupResource(), informer: f.Dataprotection().V1alpha1().Restores().Informer()}, nil
// Group=extensions.kubeblocks.io, Version=v1alpha1
case extensionsv1alpha1.SchemeGroupVersion.WithResource("addons"):
diff --git a/pkg/client/listers/dataprotection/v1alpha1/backuptool.go b/pkg/client/listers/dataprotection/v1alpha1/actionset.go
similarity index 53%
rename from pkg/client/listers/dataprotection/v1alpha1/backuptool.go
rename to pkg/client/listers/dataprotection/v1alpha1/actionset.go
index 0fbe6df4f23..c66aecad27d 100644
--- a/pkg/client/listers/dataprotection/v1alpha1/backuptool.go
+++ b/pkg/client/listers/dataprotection/v1alpha1/actionset.go
@@ -25,44 +25,44 @@ import (
"k8s.io/client-go/tools/cache"
)
-// BackupToolLister helps list BackupTools.
+// ActionSetLister helps list ActionSets.
// All objects returned here must be treated as read-only.
-type BackupToolLister interface {
- // List lists all BackupTools in the indexer.
+type ActionSetLister interface {
+ // List lists all ActionSets in the indexer.
// Objects returned here must be treated as read-only.
- List(selector labels.Selector) (ret []*v1alpha1.BackupTool, err error)
- // Get retrieves the BackupTool from the index for a given name.
+ List(selector labels.Selector) (ret []*v1alpha1.ActionSet, err error)
+ // Get retrieves the ActionSet from the index for a given name.
// Objects returned here must be treated as read-only.
- Get(name string) (*v1alpha1.BackupTool, error)
- BackupToolListerExpansion
+ Get(name string) (*v1alpha1.ActionSet, error)
+ ActionSetListerExpansion
}
-// backupToolLister implements the BackupToolLister interface.
-type backupToolLister struct {
+// actionSetLister implements the ActionSetLister interface.
+type actionSetLister struct {
indexer cache.Indexer
}
-// NewBackupToolLister returns a new BackupToolLister.
-func NewBackupToolLister(indexer cache.Indexer) BackupToolLister {
- return &backupToolLister{indexer: indexer}
+// NewActionSetLister returns a new ActionSetLister.
+func NewActionSetLister(indexer cache.Indexer) ActionSetLister {
+ return &actionSetLister{indexer: indexer}
}
-// List lists all BackupTools in the indexer.
-func (s *backupToolLister) List(selector labels.Selector) (ret []*v1alpha1.BackupTool, err error) {
+// List lists all ActionSets in the indexer.
+func (s *actionSetLister) List(selector labels.Selector) (ret []*v1alpha1.ActionSet, err error) {
err = cache.ListAll(s.indexer, selector, func(m interface{}) {
- ret = append(ret, m.(*v1alpha1.BackupTool))
+ ret = append(ret, m.(*v1alpha1.ActionSet))
})
return ret, err
}
-// Get retrieves the BackupTool from the index for a given name.
-func (s *backupToolLister) Get(name string) (*v1alpha1.BackupTool, error) {
+// Get retrieves the ActionSet from the index for a given name.
+func (s *actionSetLister) Get(name string) (*v1alpha1.ActionSet, error) {
obj, exists, err := s.indexer.GetByKey(name)
if err != nil {
return nil, err
}
if !exists {
- return nil, errors.NewNotFound(v1alpha1.Resource("backuptool"), name)
+ return nil, errors.NewNotFound(v1alpha1.Resource("actionset"), name)
}
- return obj.(*v1alpha1.BackupTool), nil
+ return obj.(*v1alpha1.ActionSet), nil
}
diff --git a/pkg/client/listers/dataprotection/v1alpha1/backupschedule.go b/pkg/client/listers/dataprotection/v1alpha1/backupschedule.go
new file mode 100644
index 00000000000..21f2fdd1a8b
--- /dev/null
+++ b/pkg/client/listers/dataprotection/v1alpha1/backupschedule.go
@@ -0,0 +1,99 @@
+/*
+Copyright (C) 2022-2023 ApeCloud Co., Ltd
+
+Licensed under the Apache License, Version 2.0 (the "License");
+you may not use this file except in compliance with the License.
+You may obtain a copy of the License at
+
+ http://www.apache.org/licenses/LICENSE-2.0
+
+Unless required by applicable law or agreed to in writing, software
+distributed under the License is distributed on an "AS IS" BASIS,
+WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+See the License for the specific language governing permissions and
+limitations under the License.
+*/
+
+// Code generated by lister-gen. DO NOT EDIT.
+
+package v1alpha1
+
+import (
+ v1alpha1 "github.com/apecloud/kubeblocks/apis/dataprotection/v1alpha1"
+ "k8s.io/apimachinery/pkg/api/errors"
+ "k8s.io/apimachinery/pkg/labels"
+ "k8s.io/client-go/tools/cache"
+)
+
+// BackupScheduleLister helps list BackupSchedules.
+// All objects returned here must be treated as read-only.
+type BackupScheduleLister interface {
+ // List lists all BackupSchedules in the indexer.
+ // Objects returned here must be treated as read-only.
+ List(selector labels.Selector) (ret []*v1alpha1.BackupSchedule, err error)
+ // BackupSchedules returns an object that can list and get BackupSchedules.
+ BackupSchedules(namespace string) BackupScheduleNamespaceLister
+ BackupScheduleListerExpansion
+}
+
+// backupScheduleLister implements the BackupScheduleLister interface.
+type backupScheduleLister struct {
+ indexer cache.Indexer
+}
+
+// NewBackupScheduleLister returns a new BackupScheduleLister.
+func NewBackupScheduleLister(indexer cache.Indexer) BackupScheduleLister {
+ return &backupScheduleLister{indexer: indexer}
+}
+
+// List lists all BackupSchedules in the indexer.
+func (s *backupScheduleLister) List(selector labels.Selector) (ret []*v1alpha1.BackupSchedule, err error) {
+ err = cache.ListAll(s.indexer, selector, func(m interface{}) {
+ ret = append(ret, m.(*v1alpha1.BackupSchedule))
+ })
+ return ret, err
+}
+
+// BackupSchedules returns an object that can list and get BackupSchedules.
+func (s *backupScheduleLister) BackupSchedules(namespace string) BackupScheduleNamespaceLister {
+ return backupScheduleNamespaceLister{indexer: s.indexer, namespace: namespace}
+}
+
+// BackupScheduleNamespaceLister helps list and get BackupSchedules.
+// All objects returned here must be treated as read-only.
+type BackupScheduleNamespaceLister interface {
+ // List lists all BackupSchedules in the indexer for a given namespace.
+ // Objects returned here must be treated as read-only.
+ List(selector labels.Selector) (ret []*v1alpha1.BackupSchedule, err error)
+ // Get retrieves the BackupSchedule from the indexer for a given namespace and name.
+ // Objects returned here must be treated as read-only.
+ Get(name string) (*v1alpha1.BackupSchedule, error)
+ BackupScheduleNamespaceListerExpansion
+}
+
+// backupScheduleNamespaceLister implements the BackupScheduleNamespaceLister
+// interface.
+type backupScheduleNamespaceLister struct {
+ indexer cache.Indexer
+ namespace string
+}
+
+// List lists all BackupSchedules in the indexer for a given namespace.
+func (s backupScheduleNamespaceLister) List(selector labels.Selector) (ret []*v1alpha1.BackupSchedule, err error) {
+ err = cache.ListAllByNamespace(s.indexer, s.namespace, selector, func(m interface{}) {
+ ret = append(ret, m.(*v1alpha1.BackupSchedule))
+ })
+ return ret, err
+}
+
+// Get retrieves the BackupSchedule from the indexer for a given namespace and name.
+func (s backupScheduleNamespaceLister) Get(name string) (*v1alpha1.BackupSchedule, error) {
+ obj, exists, err := s.indexer.GetByKey(s.namespace + "/" + name)
+ if err != nil {
+ return nil, err
+ }
+ if !exists {
+ return nil, errors.NewNotFound(v1alpha1.Resource("backupschedule"), name)
+ }
+ return obj.(*v1alpha1.BackupSchedule), nil
+}
diff --git a/pkg/client/listers/dataprotection/v1alpha1/expansion_generated.go b/pkg/client/listers/dataprotection/v1alpha1/expansion_generated.go
index e9283c098cb..8e7bab6e1c3 100644
--- a/pkg/client/listers/dataprotection/v1alpha1/expansion_generated.go
+++ b/pkg/client/listers/dataprotection/v1alpha1/expansion_generated.go
@@ -18,6 +18,10 @@ limitations under the License.
package v1alpha1
+// ActionSetListerExpansion allows custom methods to be added to
+// ActionSetLister.
+type ActionSetListerExpansion interface{}
+
// BackupListerExpansion allows custom methods to be added to
// BackupLister.
type BackupListerExpansion interface{}
@@ -38,14 +42,18 @@ type BackupPolicyNamespaceListerExpansion interface{}
// BackupRepoLister.
type BackupRepoListerExpansion interface{}
-// BackupToolListerExpansion allows custom methods to be added to
-// BackupToolLister.
-type BackupToolListerExpansion interface{}
+// BackupScheduleListerExpansion allows custom methods to be added to
+// BackupScheduleLister.
+type BackupScheduleListerExpansion interface{}
+
+// BackupScheduleNamespaceListerExpansion allows custom methods to be added to
+// BackupScheduleNamespaceLister.
+type BackupScheduleNamespaceListerExpansion interface{}
-// RestoreJobListerExpansion allows custom methods to be added to
-// RestoreJobLister.
-type RestoreJobListerExpansion interface{}
+// RestoreListerExpansion allows custom methods to be added to
+// RestoreLister.
+type RestoreListerExpansion interface{}
-// RestoreJobNamespaceListerExpansion allows custom methods to be added to
-// RestoreJobNamespaceLister.
-type RestoreJobNamespaceListerExpansion interface{}
+// RestoreNamespaceListerExpansion allows custom methods to be added to
+// RestoreNamespaceLister.
+type RestoreNamespaceListerExpansion interface{}
diff --git a/pkg/client/listers/dataprotection/v1alpha1/restore.go b/pkg/client/listers/dataprotection/v1alpha1/restore.go
new file mode 100644
index 00000000000..02979dd1108
--- /dev/null
+++ b/pkg/client/listers/dataprotection/v1alpha1/restore.go
@@ -0,0 +1,99 @@
+/*
+Copyright (C) 2022-2023 ApeCloud Co., Ltd
+
+Licensed under the Apache License, Version 2.0 (the "License");
+you may not use this file except in compliance with the License.
+You may obtain a copy of the License at
+
+ http://www.apache.org/licenses/LICENSE-2.0
+
+Unless required by applicable law or agreed to in writing, software
+distributed under the License is distributed on an "AS IS" BASIS,
+WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+See the License for the specific language governing permissions and
+limitations under the License.
+*/
+
+// Code generated by lister-gen. DO NOT EDIT.
+
+package v1alpha1
+
+import (
+ v1alpha1 "github.com/apecloud/kubeblocks/apis/dataprotection/v1alpha1"
+ "k8s.io/apimachinery/pkg/api/errors"
+ "k8s.io/apimachinery/pkg/labels"
+ "k8s.io/client-go/tools/cache"
+)
+
+// RestoreLister helps list Restores.
+// All objects returned here must be treated as read-only.
+type RestoreLister interface {
+ // List lists all Restores in the indexer.
+ // Objects returned here must be treated as read-only.
+ List(selector labels.Selector) (ret []*v1alpha1.Restore, err error)
+ // Restores returns an object that can list and get Restores.
+ Restores(namespace string) RestoreNamespaceLister
+ RestoreListerExpansion
+}
+
+// restoreLister implements the RestoreLister interface.
+type restoreLister struct {
+ indexer cache.Indexer
+}
+
+// NewRestoreLister returns a new RestoreLister.
+func NewRestoreLister(indexer cache.Indexer) RestoreLister {
+ return &restoreLister{indexer: indexer}
+}
+
+// List lists all Restores in the indexer.
+func (s *restoreLister) List(selector labels.Selector) (ret []*v1alpha1.Restore, err error) {
+ err = cache.ListAll(s.indexer, selector, func(m interface{}) {
+ ret = append(ret, m.(*v1alpha1.Restore))
+ })
+ return ret, err
+}
+
+// Restores returns an object that can list and get Restores.
+func (s *restoreLister) Restores(namespace string) RestoreNamespaceLister {
+ return restoreNamespaceLister{indexer: s.indexer, namespace: namespace}
+}
+
+// RestoreNamespaceLister helps list and get Restores.
+// All objects returned here must be treated as read-only.
+type RestoreNamespaceLister interface {
+ // List lists all Restores in the indexer for a given namespace.
+ // Objects returned here must be treated as read-only.
+ List(selector labels.Selector) (ret []*v1alpha1.Restore, err error)
+ // Get retrieves the Restore from the indexer for a given namespace and name.
+ // Objects returned here must be treated as read-only.
+ Get(name string) (*v1alpha1.Restore, error)
+ RestoreNamespaceListerExpansion
+}
+
+// restoreNamespaceLister implements the RestoreNamespaceLister
+// interface.
+type restoreNamespaceLister struct {
+ indexer cache.Indexer
+ namespace string
+}
+
+// List lists all Restores in the indexer for a given namespace.
+func (s restoreNamespaceLister) List(selector labels.Selector) (ret []*v1alpha1.Restore, err error) {
+ err = cache.ListAllByNamespace(s.indexer, s.namespace, selector, func(m interface{}) {
+ ret = append(ret, m.(*v1alpha1.Restore))
+ })
+ return ret, err
+}
+
+// Get retrieves the Restore from the indexer for a given namespace and name.
+func (s restoreNamespaceLister) Get(name string) (*v1alpha1.Restore, error) {
+ obj, exists, err := s.indexer.GetByKey(s.namespace + "/" + name)
+ if err != nil {
+ return nil, err
+ }
+ if !exists {
+ return nil, errors.NewNotFound(v1alpha1.Resource("restore"), name)
+ }
+ return obj.(*v1alpha1.Restore), nil
+}
diff --git a/pkg/client/listers/dataprotection/v1alpha1/restorejob.go b/pkg/client/listers/dataprotection/v1alpha1/restorejob.go
deleted file mode 100644
index fabe7031849..00000000000
--- a/pkg/client/listers/dataprotection/v1alpha1/restorejob.go
+++ /dev/null
@@ -1,99 +0,0 @@
-/*
-Copyright (C) 2022-2023 ApeCloud Co., Ltd
-
-Licensed under the Apache License, Version 2.0 (the "License");
-you may not use this file except in compliance with the License.
-You may obtain a copy of the License at
-
- http://www.apache.org/licenses/LICENSE-2.0
-
-Unless required by applicable law or agreed to in writing, software
-distributed under the License is distributed on an "AS IS" BASIS,
-WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
-See the License for the specific language governing permissions and
-limitations under the License.
-*/
-
-// Code generated by lister-gen. DO NOT EDIT.
-
-package v1alpha1
-
-import (
- v1alpha1 "github.com/apecloud/kubeblocks/apis/dataprotection/v1alpha1"
- "k8s.io/apimachinery/pkg/api/errors"
- "k8s.io/apimachinery/pkg/labels"
- "k8s.io/client-go/tools/cache"
-)
-
-// RestoreJobLister helps list RestoreJobs.
-// All objects returned here must be treated as read-only.
-type RestoreJobLister interface {
- // List lists all RestoreJobs in the indexer.
- // Objects returned here must be treated as read-only.
- List(selector labels.Selector) (ret []*v1alpha1.RestoreJob, err error)
- // RestoreJobs returns an object that can list and get RestoreJobs.
- RestoreJobs(namespace string) RestoreJobNamespaceLister
- RestoreJobListerExpansion
-}
-
-// restoreJobLister implements the RestoreJobLister interface.
-type restoreJobLister struct {
- indexer cache.Indexer
-}
-
-// NewRestoreJobLister returns a new RestoreJobLister.
-func NewRestoreJobLister(indexer cache.Indexer) RestoreJobLister {
- return &restoreJobLister{indexer: indexer}
-}
-
-// List lists all RestoreJobs in the indexer.
-func (s *restoreJobLister) List(selector labels.Selector) (ret []*v1alpha1.RestoreJob, err error) {
- err = cache.ListAll(s.indexer, selector, func(m interface{}) {
- ret = append(ret, m.(*v1alpha1.RestoreJob))
- })
- return ret, err
-}
-
-// RestoreJobs returns an object that can list and get RestoreJobs.
-func (s *restoreJobLister) RestoreJobs(namespace string) RestoreJobNamespaceLister {
- return restoreJobNamespaceLister{indexer: s.indexer, namespace: namespace}
-}
-
-// RestoreJobNamespaceLister helps list and get RestoreJobs.
-// All objects returned here must be treated as read-only.
-type RestoreJobNamespaceLister interface {
- // List lists all RestoreJobs in the indexer for a given namespace.
- // Objects returned here must be treated as read-only.
- List(selector labels.Selector) (ret []*v1alpha1.RestoreJob, err error)
- // Get retrieves the RestoreJob from the indexer for a given namespace and name.
- // Objects returned here must be treated as read-only.
- Get(name string) (*v1alpha1.RestoreJob, error)
- RestoreJobNamespaceListerExpansion
-}
-
-// restoreJobNamespaceLister implements the RestoreJobNamespaceLister
-// interface.
-type restoreJobNamespaceLister struct {
- indexer cache.Indexer
- namespace string
-}
-
-// List lists all RestoreJobs in the indexer for a given namespace.
-func (s restoreJobNamespaceLister) List(selector labels.Selector) (ret []*v1alpha1.RestoreJob, err error) {
- err = cache.ListAllByNamespace(s.indexer, s.namespace, selector, func(m interface{}) {
- ret = append(ret, m.(*v1alpha1.RestoreJob))
- })
- return ret, err
-}
-
-// Get retrieves the RestoreJob from the indexer for a given namespace and name.
-func (s restoreJobNamespaceLister) Get(name string) (*v1alpha1.RestoreJob, error) {
- obj, exists, err := s.indexer.GetByKey(s.namespace + "/" + name)
- if err != nil {
- return nil, err
- }
- if !exists {
- return nil, errors.NewNotFound(v1alpha1.Resource("restorejob"), name)
- }
- return obj.(*v1alpha1.RestoreJob), nil
-}
diff --git a/test/e2e/testdata/smoketest/foxlake/06_hscale_up.yaml b/test/e2e/testdata/smoketest/foxlake/06_hscale_up.yaml
index d4c8ef1a6b7..7b359fd8bf1 100644
--- a/test/e2e/testdata/smoketest/foxlake/06_hscale_up.yaml
+++ b/test/e2e/testdata/smoketest/foxlake/06_hscale_up.yaml
@@ -7,6 +7,6 @@ spec:
type: HorizontalScaling
horizontalScaling:
- componentName: foxlake-server
- replicas: 3
+ replicas: 2
- componentName: foxlake-metadb
- replicas: 3
\ No newline at end of file
+ replicas: 2
\ No newline at end of file
diff --git a/test/e2e/testdata/smoketest/oceanbase/00_oceanbasecluster.yaml b/test/e2e/testdata/smoketest/oceanbase/00_oceanbasecluster.yaml
new file mode 100644
index 00000000000..69715495e7f
--- /dev/null
+++ b/test/e2e/testdata/smoketest/oceanbase/00_oceanbasecluster.yaml
@@ -0,0 +1,95 @@
+---
+# Source: oceanbase-cluster/templates/serviceaccount.yaml
+apiVersion: v1
+kind: ServiceAccount
+metadata:
+ name: obcluster-observer-sa
+ namespace: "default"
+ labels:
+ helm.sh/chart: oceanbase-cluster-0.0.1-alpha1
+ app.kubernetes.io/name: oceanbase-cluster
+ app.kubernetes.io/instance: oceanbase-cluster
+ app.kubernetes.io/version: "4.2.0.0-100010032023083021"
+ app.kubernetes.io/managed-by: Helm
+---
+# Source: oceanbase-cluster/templates/role.yaml
+apiVersion: rbac.authorization.k8s.io/v1
+kind: Role
+metadata:
+ name: obcluster-statefulset-reader
+ namespace: "default"
+ labels:
+ helm.sh/chart: oceanbase-cluster-0.0.1-alpha1
+ app.kubernetes.io/name: oceanbase-cluster
+ app.kubernetes.io/instance: oceanbase-cluster
+ app.kubernetes.io/version: "4.2.0.0-100010032023083021"
+ app.kubernetes.io/managed-by: Helm
+rules:
+- apiGroups: ["apps"] # "" indicates the core API group
+ resources: ["statefulsets"]
+ verbs: ["get", "watch", "list"]
+---
+# Source: oceanbase-cluster/templates/rolebinding.yaml
+apiVersion: rbac.authorization.k8s.io/v1
+kind: RoleBinding
+metadata:
+ name: obcluster-read-statefulsets
+ namespace: "default"
+ labels:
+ helm.sh/chart: oceanbase-cluster-0.0.1-alpha1
+ app.kubernetes.io/name: oceanbase-cluster
+ app.kubernetes.io/instance: oceanbase-cluster
+ app.kubernetes.io/version: "4.2.0.0-100010032023083021"
+ app.kubernetes.io/managed-by: Helm
+subjects:
+- kind: ServiceAccount
+ name: obcluster-observer-sa
+- kind: ServiceAccount
+ name: kb-obcluster
+roleRef:
+ kind: Role
+ name: obcluster-statefulset-reader
+ apiGroup: rbac.authorization.k8s.io
+---
+# Source: oceanbase-cluster/templates/cluster.yaml
+apiVersion: apps.kubeblocks.io/v1alpha1
+kind: Cluster
+metadata:
+ name: oceanbase-cluster
+ labels:
+ helm.sh/chart: oceanbase-cluster-0.0.1-alpha1
+ app.kubernetes.io/name: oceanbase-cluster
+ app.kubernetes.io/instance: oceanbase-cluster
+ app.kubernetes.io/version: "4.2.0.0-100010032023083021"
+ app.kubernetes.io/managed-by: Helm
+spec:
+ clusterDefinitionRef: oceanbase
+ clusterVersionRef: oceanbase-4.2.0.0-100010032023083021
+ terminationPolicy: Delete
+ componentSpecs:
+ - name: ob-bundle
+ componentDefRef: ob-bundle
+ serviceAccountName: obcluster-observer-sa
+ replicas: 3
+ volumeClaimTemplates:
+ - name: data-file
+ spec:
+ accessModes:
+ - ReadWriteOnce
+ resources:
+ requests:
+ storage: "50Gi"
+ - name: data-log
+ spec:
+ accessModes:
+ - ReadWriteOnce
+ resources:
+ requests:
+ storage: "50Gi"
+ - name: log
+ spec:
+ accessModes:
+ - ReadWriteOnce
+ resources:
+ requests:
+ storage: "20Gi"
diff --git a/test/e2e/testdata/smoketest/oceanbase/01_vscale.yaml b/test/e2e/testdata/smoketest/oceanbase/01_vscale.yaml
new file mode 100644
index 00000000000..a7ed2425d20
--- /dev/null
+++ b/test/e2e/testdata/smoketest/oceanbase/01_vscale.yaml
@@ -0,0 +1,12 @@
+apiVersion: apps.kubeblocks.io/v1alpha1
+kind: OpsRequest
+metadata:
+ generateName: oceanbase-cluster-vscale-
+spec:
+ clusterRef: oceanbase-cluster
+ type: VerticalScaling
+ verticalScaling:
+ - componentName: ob-bundle
+ requests:
+ cpu: "1.5"
+ memory: 1Gi
\ No newline at end of file
diff --git a/test/e2e/testdata/smoketest/oceanbase/02_stop.yaml b/test/e2e/testdata/smoketest/oceanbase/02_stop.yaml
new file mode 100644
index 00000000000..9d750d7cfe9
--- /dev/null
+++ b/test/e2e/testdata/smoketest/oceanbase/02_stop.yaml
@@ -0,0 +1,10 @@
+apiVersion: apps.kubeblocks.io/v1alpha1
+kind: OpsRequest
+metadata:
+ generateName: oceanbase-cluster-stop-
+spec:
+ clusterRef: oceanbase-cluster
+ ttlSecondsAfterSucceed: 27017
+ type: Stop
+ restart:
+ - componentName: ob-bundle
\ No newline at end of file
diff --git a/test/e2e/testdata/smoketest/oceanbase/03_start.yaml b/test/e2e/testdata/smoketest/oceanbase/03_start.yaml
new file mode 100644
index 00000000000..49c1b14ae07
--- /dev/null
+++ b/test/e2e/testdata/smoketest/oceanbase/03_start.yaml
@@ -0,0 +1,10 @@
+apiVersion: apps.kubeblocks.io/v1alpha1
+kind: OpsRequest
+metadata:
+ generateName: oceanbase-cluster-start-
+spec:
+ clusterRef: oceanbase-cluster
+ ttlSecondsAfterSucceed: 27017
+ type: Start
+ restart:
+ - componentName: ob-bundle
\ No newline at end of file
diff --git a/test/e2e/testdata/smoketest/oceanbase/04_vexpand.yaml b/test/e2e/testdata/smoketest/oceanbase/04_vexpand.yaml
new file mode 100644
index 00000000000..b0af0ba1a5c
--- /dev/null
+++ b/test/e2e/testdata/smoketest/oceanbase/04_vexpand.yaml
@@ -0,0 +1,16 @@
+apiVersion: apps.kubeblocks.io/v1alpha1
+kind: OpsRequest
+metadata:
+ generateName: oceanbase-cluster-vexpand-
+spec:
+ clusterRef: oceanbase-cluster
+ type: VolumeExpansion
+ volumeExpansion:
+ - componentName: ob-bundle
+ volumeClaimTemplates:
+ - name: data-file
+ storage: "51Gi"
+ - name: data-log
+ storage: "51Gi"
+ - name: log
+ storage: "51Gi"
\ No newline at end of file
diff --git a/test/e2e/testdata/smoketest/oceanbase/05_restart.yaml b/test/e2e/testdata/smoketest/oceanbase/05_restart.yaml
new file mode 100644
index 00000000000..08d3bc9dfaf
--- /dev/null
+++ b/test/e2e/testdata/smoketest/oceanbase/05_restart.yaml
@@ -0,0 +1,10 @@
+apiVersion: apps.kubeblocks.io/v1alpha1
+kind: OpsRequest
+metadata:
+ generateName: oceanbase-cluster-restart-
+spec:
+ clusterRef: oceanbase-cluster
+ ttlSecondsAfterSucceed: 27017
+ type: Restart
+ restart:
+ - componentName: ob-bundle
\ No newline at end of file
diff --git a/test/e2e/testdata/smoketest/oceanbase/06_hscale_up.yaml b/test/e2e/testdata/smoketest/oceanbase/06_hscale_up.yaml
new file mode 100644
index 00000000000..c4f14b45fe9
--- /dev/null
+++ b/test/e2e/testdata/smoketest/oceanbase/06_hscale_up.yaml
@@ -0,0 +1,10 @@
+apiVersion: apps.kubeblocks.io/v1alpha1
+kind: OpsRequest
+metadata:
+ generateName: oceanbase-cluster-hscale-up-
+spec:
+ clusterRef: oceanbase-cluster
+ type: HorizontalScaling
+ horizontalScaling:
+ - componentName: ob-bundle
+ replicas: 4
\ No newline at end of file
diff --git a/test/e2e/testdata/smoketest/oceanbase/07_hscale_down.yaml b/test/e2e/testdata/smoketest/oceanbase/07_hscale_down.yaml
new file mode 100644
index 00000000000..ad50ee949d7
--- /dev/null
+++ b/test/e2e/testdata/smoketest/oceanbase/07_hscale_down.yaml
@@ -0,0 +1,10 @@
+apiVersion: apps.kubeblocks.io/v1alpha1
+kind: OpsRequest
+metadata:
+ generateName: oceanbase-cluster-hscale-down-
+spec:
+ clusterRef: oceanbase-cluster
+ type: HorizontalScaling
+ horizontalScaling:
+ - componentName: ob-bundle
+ replicas: 2
\ No newline at end of file
diff --git a/test/e2e/testdata/smoketest/official-postgresql/00_official_pgcluster.yaml b/test/e2e/testdata/smoketest/official-postgresql/00_official_pgcluster.yaml
new file mode 100644
index 00000000000..c55ece3d8fd
--- /dev/null
+++ b/test/e2e/testdata/smoketest/official-postgresql/00_official_pgcluster.yaml
@@ -0,0 +1,91 @@
+---
+# Source: official-postgresql-cluster/templates/rbac.yaml
+apiVersion: v1
+kind: ServiceAccount
+metadata:
+ name: kb-official-pg
+ namespace: default
+ labels:
+ helm.sh/chart: official-postgresql-cluster-0.7.0-alpha.0
+ app.kubernetes.io/version: "14.7"
+ app.kubernetes.io/instance: official-pg
+---
+# Source: official-postgresql-cluster/templates/rbac.yaml
+apiVersion: rbac.authorization.k8s.io/v1
+kind: ClusterRoleBinding
+metadata:
+ name: kb-official-pg
+ labels:
+ helm.sh/chart: official-postgresql-cluster-0.7.0-alpha.0
+ app.kubernetes.io/version: "14.7"
+ app.kubernetes.io/instance: official-pg
+roleRef:
+ apiGroup: rbac.authorization.k8s.io
+ kind: ClusterRole
+ name: kubeblocks-volume-protection-pod-role
+subjects:
+ - kind: ServiceAccount
+ name: kb-official-pg
+ namespace: default
+---
+# Source: official-postgresql-cluster/templates/rbac.yaml
+apiVersion: rbac.authorization.k8s.io/v1
+kind: RoleBinding
+metadata:
+ name: kb-official-pg
+ labels:
+ helm.sh/chart: official-postgresql-cluster-0.7.0-alpha.0
+ app.kubernetes.io/version: "14.7"
+ app.kubernetes.io/instance: official-pg
+roleRef:
+ apiGroup: rbac.authorization.k8s.io
+ kind: ClusterRole
+ name: kubeblocks-cluster-pod-role
+subjects:
+ - kind: ServiceAccount
+ name: kb-official-pg
+ namespace: default
+---
+# Source: official-postgresql-cluster/templates/cluster.yaml
+apiVersion: apps.kubeblocks.io/v1alpha1
+kind: Cluster
+metadata:
+ name: official-pg
+ namespace: default
+ labels:
+ helm.sh/chart: official-postgresql-cluster-0.7.0-alpha.0
+ app.kubernetes.io/version: "14.7"
+ app.kubernetes.io/instance: official-pg
+spec:
+ clusterVersionRef: official-postgresql-14.7
+ terminationPolicy: Delete
+ affinity:
+ podAntiAffinity: Preferred
+ topologyKeys:
+ - kubernetes.io/hostname
+ tenancy: SharedNode
+ clusterDefinitionRef: official-postgresql
+ componentSpecs:
+ - name: postgresql
+ componentDefRef: postgresql
+ monitor: false
+ replicas: 1
+ serviceAccountName: kb-official-pg
+ switchPolicy:
+ type: Noop
+ resources:
+ limits:
+ cpu: "0.5"
+ memory: "0.5Gi"
+ requests:
+ cpu: "0.5"
+ memory: "0.5Gi"
+ volumeClaimTemplates:
+ - name: data # ref clusterDefinition components.containers.volumeMounts.name
+ spec:
+ accessModes:
+ - ReadWriteOnce
+ resources:
+ requests:
+ storage: 20Gi
+ services:
diff --git a/test/e2e/testdata/smoketest/official-postgresql/01_vscale.yaml b/test/e2e/testdata/smoketest/official-postgresql/01_vscale.yaml
new file mode 100644
index 00000000000..dcc21936420
--- /dev/null
+++ b/test/e2e/testdata/smoketest/official-postgresql/01_vscale.yaml
@@ -0,0 +1,12 @@
+apiVersion: apps.kubeblocks.io/v1alpha1
+kind: OpsRequest
+metadata:
+ generateName: official-pg-vscale-
+spec:
+ clusterRef: official-pg
+ type: VerticalScaling
+ verticalScaling:
+ - componentName: postgresql
+ requests:
+ cpu: "1"
+ memory: 1Gi
\ No newline at end of file
diff --git a/test/e2e/testdata/smoketest/official-postgresql/02_stop.yaml b/test/e2e/testdata/smoketest/official-postgresql/02_stop.yaml
new file mode 100644
index 00000000000..e875d772f0d
--- /dev/null
+++ b/test/e2e/testdata/smoketest/official-postgresql/02_stop.yaml
@@ -0,0 +1,10 @@
+apiVersion: apps.kubeblocks.io/v1alpha1
+kind: OpsRequest
+metadata:
+ generateName: official-pg-stop-
+spec:
+ clusterRef: official-pg
+ ttlSecondsAfterSucceed: 27017
+ type: Stop
+ restart:
+ - componentName: postgresql
\ No newline at end of file
diff --git a/test/e2e/testdata/smoketest/official-postgresql/03_start.yaml b/test/e2e/testdata/smoketest/official-postgresql/03_start.yaml
new file mode 100644
index 00000000000..79f0aa585ea
--- /dev/null
+++ b/test/e2e/testdata/smoketest/official-postgresql/03_start.yaml
@@ -0,0 +1,10 @@
+apiVersion: apps.kubeblocks.io/v1alpha1
+kind: OpsRequest
+metadata:
+ generateName: official-pg-start-
+spec:
+ clusterRef: official-pg
+ ttlSecondsAfterSucceed: 27017
+ type: Start
+ restart:
+ - componentName: postgresql
\ No newline at end of file
diff --git a/test/e2e/testdata/smoketest/official-postgresql/04_vexpand.yaml b/test/e2e/testdata/smoketest/official-postgresql/04_vexpand.yaml
new file mode 100644
index 00000000000..3da723e84ac
--- /dev/null
+++ b/test/e2e/testdata/smoketest/official-postgresql/04_vexpand.yaml
@@ -0,0 +1,12 @@
+apiVersion: apps.kubeblocks.io/v1alpha1
+kind: OpsRequest
+metadata:
+ generateName: official-pg-vexpand-
+spec:
+ clusterRef: official-pg
+ type: VolumeExpansion
+ volumeExpansion:
+ - componentName: postgresql
+ volumeClaimTemplates:
+ - name: data
+ storage: "21Gi"
\ No newline at end of file
diff --git a/test/e2e/testdata/smoketest/official-postgresql/05_restart.yaml b/test/e2e/testdata/smoketest/official-postgresql/05_restart.yaml
new file mode 100644
index 00000000000..4bb8b62cf11
--- /dev/null
+++ b/test/e2e/testdata/smoketest/official-postgresql/05_restart.yaml
@@ -0,0 +1,10 @@
+apiVersion: apps.kubeblocks.io/v1alpha1
+kind: OpsRequest
+metadata:
+ generateName: official-pg-restart-
+spec:
+ clusterRef: official-pg
+ ttlSecondsAfterSucceed: 27017
+ type: Restart
+ restart:
+ - componentName: postgresql
\ No newline at end of file
diff --git a/test/e2e/testdata/smoketest/official-postgresql/06_hscale_up.yaml b/test/e2e/testdata/smoketest/official-postgresql/06_hscale_up.yaml
new file mode 100644
index 00000000000..ea7b6a98d22
--- /dev/null
+++ b/test/e2e/testdata/smoketest/official-postgresql/06_hscale_up.yaml
@@ -0,0 +1,10 @@
+apiVersion: apps.kubeblocks.io/v1alpha1
+kind: OpsRequest
+metadata:
+ generateName: official-pg-hscale-up-
+spec:
+ clusterRef: official-pg
+ type: HorizontalScaling
+ horizontalScaling:
+ - componentName: postgresql
+ replicas: 3
\ No newline at end of file
diff --git a/test/e2e/testdata/smoketest/official-postgresql/07_hscale_down.yaml b/test/e2e/testdata/smoketest/official-postgresql/07_hscale_down.yaml
new file mode 100644
index 00000000000..53d15ba74a9
--- /dev/null
+++ b/test/e2e/testdata/smoketest/official-postgresql/07_hscale_down.yaml
@@ -0,0 +1,10 @@
+apiVersion: apps.kubeblocks.io/v1alpha1
+kind: OpsRequest
+metadata:
+ generateName: official-pg-hscale-down-
+spec:
+ clusterRef: official-pg
+ type: HorizontalScaling
+ horizontalScaling:
+ - componentName: postgresql
+ replicas: 2
\ No newline at end of file
diff --git a/test/e2e/testdata/smoketest/openldap/00_openldapcluster.yaml b/test/e2e/testdata/smoketest/openldap/00_openldapcluster.yaml
new file mode 100644
index 00000000000..6c9e7a2b25d
--- /dev/null
+++ b/test/e2e/testdata/smoketest/openldap/00_openldapcluster.yaml
@@ -0,0 +1,67 @@
+---
+# Source: openldap-cluster/templates/serviceaccount.yaml
+apiVersion: v1
+kind: ServiceAccount
+metadata:
+ name: openldap-cluster
+ labels:
+ helm.sh/chart: openldap-cluster-0.1.0-alpha.0
+ app.kubernetes.io/name: openldap-cluster
+ app.kubernetes.io/instance: openldap-cluster
+ app.kubernetes.io/version: "2.4.57"
+ app.kubernetes.io/managed-by: Helm
+---
+# Source: openldap-cluster/templates/cluster.yaml
+apiVersion: apps.kubeblocks.io/v1alpha1
+kind: Cluster
+metadata:
+ name: openldap-cluster
+ labels:
+ helm.sh/chart: openldap-cluster-0.1.0-alpha.0
+ app.kubernetes.io/name: openldap-cluster
+ app.kubernetes.io/instance: openldap-cluster
+ app.kubernetes.io/version: "2.4.57"
+ app.kubernetes.io/managed-by: Helm
+spec:
+ clusterDefinitionRef: openldap
+ clusterVersionRef: openldap-2.4.57
+ terminationPolicy: Halt
+ affinity:
+ topologyKeys:
+ - kubernetes.io/hostname
+ componentSpecs:
+ - name: openldap
+ componentDefRef: openldap-compdef
+ replicas: 1
+ serviceAccountName: openldap-cluster
+---
+# Source: openldap-cluster/templates/tests/test-connection.yaml
+apiVersion: v1
+kind: Pod
+metadata:
+ name: "openldap-cluster-test-connection"
+ labels:
+ helm.sh/chart: openldap-cluster-0.1.0-alpha.0
+ app.kubernetes.io/name: openldap-cluster
+ app.kubernetes.io/instance: openldap-cluster
+ app.kubernetes.io/version: "2.4.57"
+ app.kubernetes.io/managed-by: Helm
+ annotations:
+ "helm.sh/hook": test
+spec:
+ containers:
+ - name: test-openldap-cluster
+ image: "docker.io/osixia/openldap:1.5.0"
+ command:
+ - "ldapsearch"
+ args:
+ - "-x"
+ - "-H"
+ - "ldap://openldap-cluster-openldap-0.openldap-cluster-openldap-headless.default.svc.cluster.local"
+ - "-b"
+ - "dc=kubeblocks,dc=io"
+ - "-D"
+ - "cn=admin,dc=kubeblocks,dc=io"
+ - "-w"
+ - "admin"
+ restartPolicy: Never
diff --git a/test/e2e/testdata/smoketest/openldap/01_vscale.yaml b/test/e2e/testdata/smoketest/openldap/01_vscale.yaml
new file mode 100644
index 00000000000..38917694ad7
--- /dev/null
+++ b/test/e2e/testdata/smoketest/openldap/01_vscale.yaml
@@ -0,0 +1,12 @@
+apiVersion: apps.kubeblocks.io/v1alpha1
+kind: OpsRequest
+metadata:
+ generateName: openldap-cluster-vscale-
+spec:
+ clusterRef: openldap-cluster
+ type: VerticalScaling
+ verticalScaling:
+ - componentName: openldap
+ requests:
+ cpu: "1"
+ memory: 1Gi
\ No newline at end of file
diff --git a/test/e2e/testdata/smoketest/openldap/02_stop.yaml b/test/e2e/testdata/smoketest/openldap/02_stop.yaml
new file mode 100644
index 00000000000..bfb3e822a46
--- /dev/null
+++ b/test/e2e/testdata/smoketest/openldap/02_stop.yaml
@@ -0,0 +1,10 @@
+apiVersion: apps.kubeblocks.io/v1alpha1
+kind: OpsRequest
+metadata:
+ generateName: openldap-cluster-stop-
+spec:
+ clusterRef: openldap-cluster
+ ttlSecondsAfterSucceed: 27017
+ type: Stop
+ restart:
+ - componentName: openldap
\ No newline at end of file
diff --git a/test/e2e/testdata/smoketest/openldap/03_start.yaml b/test/e2e/testdata/smoketest/openldap/03_start.yaml
new file mode 100644
index 00000000000..f364fbbb7c6
--- /dev/null
+++ b/test/e2e/testdata/smoketest/openldap/03_start.yaml
@@ -0,0 +1,10 @@
+apiVersion: apps.kubeblocks.io/v1alpha1
+kind: OpsRequest
+metadata:
+ generateName: openldap-cluster-start-
+spec:
+ clusterRef: openldap-cluster
+ ttlSecondsAfterSucceed: 27017
+ type: Start
+ restart:
+ - componentName: openldap
\ No newline at end of file
diff --git a/test/e2e/testdata/smoketest/openldap/04_restart.yaml b/test/e2e/testdata/smoketest/openldap/04_restart.yaml
new file mode 100644
index 00000000000..5299247a7fc
--- /dev/null
+++ b/test/e2e/testdata/smoketest/openldap/04_restart.yaml
@@ -0,0 +1,10 @@
+apiVersion: apps.kubeblocks.io/v1alpha1
+kind: OpsRequest
+metadata:
+ generateName: openldap-cluster-restart-
+spec:
+ clusterRef: openldap-cluster
+ ttlSecondsAfterSucceed: 27017
+ type: Restart
+ restart:
+ - componentName: openldap
\ No newline at end of file
diff --git a/test/e2e/testdata/smoketest/openldap/05_hscale_up.yaml b/test/e2e/testdata/smoketest/openldap/05_hscale_up.yaml
new file mode 100644
index 00000000000..56d637c2993
--- /dev/null
+++ b/test/e2e/testdata/smoketest/openldap/05_hscale_up.yaml
@@ -0,0 +1,10 @@
+apiVersion: apps.kubeblocks.io/v1alpha1
+kind: OpsRequest
+metadata:
+ generateName: openldap-cluster-hscale-up-
+spec:
+ clusterRef: openldap-cluster
+ type: HorizontalScaling
+ horizontalScaling:
+ - componentName: openldap
+ replicas: 3
\ No newline at end of file
diff --git a/test/e2e/testdata/smoketest/openldap/06_hscale_down.yaml b/test/e2e/testdata/smoketest/openldap/06_hscale_down.yaml
new file mode 100644
index 00000000000..87082b7829b
--- /dev/null
+++ b/test/e2e/testdata/smoketest/openldap/06_hscale_down.yaml
@@ -0,0 +1,10 @@
+apiVersion: apps.kubeblocks.io/v1alpha1
+kind: OpsRequest
+metadata:
+ generateName: openldap-cluster-hscale-down-
+spec:
+ clusterRef: openldap-cluster
+ type: HorizontalScaling
+ horizontalScaling:
+ - componentName: openldap
+ replicas: 2
\ No newline at end of file
diff --git a/test/e2e/testdata/smoketest/orioledb/00_orioledbcluster.yaml b/test/e2e/testdata/smoketest/orioledb/00_orioledbcluster.yaml
new file mode 100644
index 00000000000..eed85e5b1d8
--- /dev/null
+++ b/test/e2e/testdata/smoketest/orioledb/00_orioledbcluster.yaml
@@ -0,0 +1,93 @@
+---
+# Source: orioledb-cluster/templates/rbac.yaml
+apiVersion: v1
+kind: ServiceAccount
+metadata:
+ name: kb-oriole-cluster
+ namespace: default
+ labels:
+ helm.sh/chart: orioledb-cluster-0.6.0-beta.44
+ app.kubernetes.io/version: "14.7.2-beta1"
+ app.kubernetes.io/instance: oriole-cluster
+---
+# Source: orioledb-cluster/templates/rbac.yaml
+apiVersion: rbac.authorization.k8s.io/v1
+kind: ClusterRoleBinding
+metadata:
+ name: kb-oriole-cluster
+ labels:
+ helm.sh/chart: orioledb-cluster-0.6.0-beta.44
+ app.kubernetes.io/version: "14.7.2-beta1"
+ app.kubernetes.io/instance: oriole-cluster
+roleRef:
+ apiGroup: rbac.authorization.k8s.io
+ kind: ClusterRole
+ name: kubeblocks-volume-protection-pod-role
+subjects:
+ - kind: ServiceAccount
+ name: kb-oriole-cluster
+ namespace: default
+---
+# Source: orioledb-cluster/templates/rbac.yaml
+apiVersion: rbac.authorization.k8s.io/v1
+kind: RoleBinding
+metadata:
+ name: kb-oriole-cluster
+ labels:
+ helm.sh/chart: orioledb-cluster-0.6.0-beta.44
+ app.kubernetes.io/version: "14.7.2-beta1"
+ app.kubernetes.io/instance: oriole-cluster
+roleRef:
+ apiGroup: rbac.authorization.k8s.io
+ kind: ClusterRole
+ name: kubeblocks-cluster-pod-role
+subjects:
+ - kind: ServiceAccount
+ name: kb-oriole-cluster
+ namespace: default
+---
+# Source: orioledb-cluster/templates/cluster.yaml
+apiVersion: apps.kubeblocks.io/v1alpha1
+kind: Cluster
+metadata:
+ name: oriole-cluster
+ namespace: default
+ labels:
+ helm.sh/chart: orioledb-cluster-0.6.0-beta.44
+ app.kubernetes.io/version: "14.7.2-beta1"
+ app.kubernetes.io/instance: oriole-cluster
+spec:
+ clusterVersionRef: orioledb-beta1
+ terminationPolicy: Delete
+ affinity:
+ podAntiAffinity: Preferred
+ topologyKeys:
+ - kubernetes.io/hostname
+ tenancy: SharedNode
+ clusterDefinitionRef: orioledb
+ componentSpecs:
+ - name: orioledb
+ componentDefRef: orioledb
+ monitor: false
+ replicas: 1
+ enabledLogs:
+ - running
+ serviceAccountName: kb-oriole-cluster
+ switchPolicy:
+ type: Noop
+ resources:
+ limits:
+ cpu: "0.5"
+ memory: "0.5Gi"
+ requests:
+ cpu: "0.5"
+ memory: "0.5Gi"
+ volumeClaimTemplates:
+ - name: data # ref clusterDefinition components.containers.volumeMounts.name
+ spec:
+ accessModes:
+ - ReadWriteOnce
+ resources:
+ requests:
+ storage: 20Gi
+ services:
diff --git a/test/e2e/testdata/smoketest/orioledb/01_vscale.yaml b/test/e2e/testdata/smoketest/orioledb/01_vscale.yaml
new file mode 100644
index 00000000000..1acc234a06e
--- /dev/null
+++ b/test/e2e/testdata/smoketest/orioledb/01_vscale.yaml
@@ -0,0 +1,12 @@
+apiVersion: apps.kubeblocks.io/v1alpha1
+kind: OpsRequest
+metadata:
+ generateName: oriole-cluster-vscale-
+spec:
+ clusterRef: oriole-cluster
+ type: VerticalScaling
+ verticalScaling:
+ - componentName: orioledb
+ requests:
+ cpu: "1"
+ memory: 1Gi
\ No newline at end of file
diff --git a/test/e2e/testdata/smoketest/orioledb/02_stop.yaml b/test/e2e/testdata/smoketest/orioledb/02_stop.yaml
new file mode 100644
index 00000000000..71c863d73d2
--- /dev/null
+++ b/test/e2e/testdata/smoketest/orioledb/02_stop.yaml
@@ -0,0 +1,10 @@
+apiVersion: apps.kubeblocks.io/v1alpha1
+kind: OpsRequest
+metadata:
+ generateName: oriole-cluster-stop-
+spec:
+ clusterRef: oriole-cluster
+ ttlSecondsAfterSucceed: 27017
+ type: Stop
+ restart:
+ - componentName: orioledb
\ No newline at end of file
diff --git a/test/e2e/testdata/smoketest/orioledb/03_start.yaml b/test/e2e/testdata/smoketest/orioledb/03_start.yaml
new file mode 100644
index 00000000000..e0c3f8a79a8
--- /dev/null
+++ b/test/e2e/testdata/smoketest/orioledb/03_start.yaml
@@ -0,0 +1,10 @@
+apiVersion: apps.kubeblocks.io/v1alpha1
+kind: OpsRequest
+metadata:
+ generateName: oriole-cluster-start-
+spec:
+ clusterRef: oriole-cluster
+ ttlSecondsAfterSucceed: 27017
+ type: Start
+ restart:
+ - componentName: orioledb
\ No newline at end of file
diff --git a/test/e2e/testdata/smoketest/orioledb/04_vexpand.yaml b/test/e2e/testdata/smoketest/orioledb/04_vexpand.yaml
new file mode 100644
index 00000000000..52ccab7c83d
--- /dev/null
+++ b/test/e2e/testdata/smoketest/orioledb/04_vexpand.yaml
@@ -0,0 +1,12 @@
+apiVersion: apps.kubeblocks.io/v1alpha1
+kind: OpsRequest
+metadata:
+ generateName: oriole-cluster-vexpand-
+spec:
+ clusterRef: oriole-cluster
+ type: VolumeExpansion
+ volumeExpansion:
+ - componentName: orioledb
+ volumeClaimTemplates:
+ - name: data
+ storage: "21Gi"
\ No newline at end of file
diff --git a/test/e2e/testdata/smoketest/orioledb/05_restart.yaml b/test/e2e/testdata/smoketest/orioledb/05_restart.yaml
new file mode 100644
index 00000000000..fe12e90fb48
--- /dev/null
+++ b/test/e2e/testdata/smoketest/orioledb/05_restart.yaml
@@ -0,0 +1,10 @@
+apiVersion: apps.kubeblocks.io/v1alpha1
+kind: OpsRequest
+metadata:
+ generateName: oriole-cluster-restart-
+spec:
+ clusterRef: oriole-cluster
+ ttlSecondsAfterSucceed: 27017
+ type: Restart
+ restart:
+ - componentName: orioledb
\ No newline at end of file
diff --git a/test/e2e/testdata/smoketest/orioledb/06_hscale_up.yaml b/test/e2e/testdata/smoketest/orioledb/06_hscale_up.yaml
new file mode 100644
index 00000000000..52972bca93b
--- /dev/null
+++ b/test/e2e/testdata/smoketest/orioledb/06_hscale_up.yaml
@@ -0,0 +1,10 @@
+apiVersion: apps.kubeblocks.io/v1alpha1
+kind: OpsRequest
+metadata:
+ generateName: orioledb-cluster-hscale-up-
+spec:
+ clusterRef: orioledb-cluster
+ type: HorizontalScaling
+ horizontalScaling:
+ - componentName: orioledb
+ replicas: 3
\ No newline at end of file
diff --git a/test/e2e/testdata/smoketest/orioledb/07_hscale_down.yaml b/test/e2e/testdata/smoketest/orioledb/07_hscale_down.yaml
new file mode 100644
index 00000000000..5e74ad073de
--- /dev/null
+++ b/test/e2e/testdata/smoketest/orioledb/07_hscale_down.yaml
@@ -0,0 +1,10 @@
+apiVersion: apps.kubeblocks.io/v1alpha1
+kind: OpsRequest
+metadata:
+ generateName: orioledb-cluster-hscale-down-
+spec:
+ clusterRef: orioledb-cluster
+ type: HorizontalScaling
+ horizontalScaling:
+ - componentName: orioledb
+ replicas: 2
\ No newline at end of file
diff --git a/test/integration/backup_mysql_test.go b/test/integration/backup_mysql_test.go
index 750394e3415..3d2dc5a82f0 100644
--- a/test/integration/backup_mysql_test.go
+++ b/test/integration/backup_mysql_test.go
@@ -30,6 +30,7 @@ import (
"github.com/apecloud/kubeblocks/internal/controller/component"
intctrlutil "github.com/apecloud/kubeblocks/internal/generics"
testapps "github.com/apecloud/kubeblocks/internal/testutil/apps"
+ testdp "github.com/apecloud/kubeblocks/internal/testutil/dataprotection"
)
var _ = Describe("MySQL data protection function", func() {
@@ -63,8 +64,8 @@ var _ = Describe("MySQL data protection function", func() {
testapps.ClearResources(&testCtx, intctrlutil.ConfigMapSignature, inNS, ml)
testapps.ClearResourcesWithRemoveFinalizerOption(&testCtx, intctrlutil.BackupSignature, true, inNS)
testapps.ClearResources(&testCtx, intctrlutil.BackupPolicySignature, inNS, ml)
- testapps.ClearResources(&testCtx, intctrlutil.BackupToolSignature, inNS, ml)
- testapps.ClearResources(&testCtx, intctrlutil.RestoreJobSignature, inNS, ml)
+ testapps.ClearResources(&testCtx, intctrlutil.ActionSetSignature, inNS, ml)
+ testapps.ClearResources(&testCtx, intctrlutil.RestoreSignature, inNS, ml)
}
@@ -100,7 +101,6 @@ var _ = Describe("MySQL data protection function", func() {
Create(&testCtx).GetObject()
By("Create a cluster obj")
-
pvcSpec := testapps.NewPVCSpec("1Gi")
clusterObj = testapps.NewClusterFactory(testCtx.DefaultNamespace, clusterNamePrefix,
clusterDefObj.Name, clusterVersionObj.Name).WithRandomName().
@@ -118,18 +118,17 @@ var _ = Describe("MySQL data protection function", func() {
}
createBackupObj := func() {
- By("By creating a backupTool")
- backupTool := testapps.CreateCustomizedObj(&testCtx, "backup/backuptool.yaml",
- &dpv1alpha1.BackupTool{}, testapps.RandomizedObjName())
+ By("By creating a actionSet")
+ actionSet := testapps.CreateCustomizedObj(&testCtx, "backup/actionset.yaml",
+ &dpv1alpha1.ActionSet{}, testapps.RandomizedObjName())
By("By creating a backupPolicy from backupPolicyTemplate: " + backupPolicyTemplateName)
- backupPolicyObj := testapps.NewBackupPolicyFactory(testCtx.DefaultNamespace, backupPolicyName).
+ backupPolicyObj := testdp.NewBackupPolicyFactory(testCtx.DefaultNamespace, backupPolicyName).
WithRandomName().
- AddDataFilePolicy().
- SetBackupToolName(backupTool.Name).
- AddMatchLabels(constant.AppInstanceLabelKey, clusterKey.Name).
- SetTargetSecretName(component.GenerateConnCredential(clusterKey.Name)).
- SetPVC(backupRemotePVCName).
+ SetTarget(constant.AppInstanceLabelKey, clusterKey.Name).
+ SetTargetConnectionCredential(component.GenerateConnCredential(clusterKey.Name)).
+ AddBackupMethod(testdp.BackupMethodName, false, actionSet.Name).
+ SetBackupMethodVolumeMounts(testapps.DataVolumeName, "/data").
Create(&testCtx).GetObject()
backupPolicyKey := client.ObjectKeyFromObject(backupPolicyObj)
@@ -142,14 +141,14 @@ var _ = Describe("MySQL data protection function", func() {
By("By check backupPolicy available")
Eventually(testapps.CheckObj(&testCtx, backupPolicyKey, func(g Gomega, backupPolicy *dpv1alpha1.BackupPolicy) {
- g.Expect(backupPolicy.Status.Phase).To(Equal(dpv1alpha1.PolicyAvailable))
+ g.Expect(backupPolicy.Status.Phase).To(Equal(dpv1alpha1.BackupPolicyAvailable))
})).Should(Succeed())
By("By creating a backup from backupPolicy: " + backupPolicyKey.Name)
- backup := testapps.NewBackupFactory(testCtx.DefaultNamespace, backupName).
+ backup := testdp.NewBackupFactory(testCtx.DefaultNamespace, backupName).
WithRandomName().
SetBackupPolicyName(backupPolicyKey.Name).
- SetBackupType(dpv1alpha1.BackupTypeDataFile).
+ SetBackupMethod(testdp.BackupMethodName).
Create(&testCtx).GetObject()
backupKey = client.ObjectKeyFromObject(backup)
}
@@ -162,7 +161,7 @@ var _ = Describe("MySQL data protection function", func() {
It("should be completed", func() {
Eventually(testapps.CheckObj(&testCtx, backupKey, func(g Gomega, backup *dpv1alpha1.Backup) {
- g.Expect(backup.Status.Phase).To(Equal(dpv1alpha1.BackupCompleted))
+ g.Expect(backup.Status.Phase).To(Equal(dpv1alpha1.BackupPhaseCompleted))
})).Should(Succeed())
})
})
diff --git a/test/integration/controller_suite_test.go b/test/integration/controller_suite_test.go
index 3ca98b67867..65ca963f1c8 100644
--- a/test/integration/controller_suite_test.go
+++ b/test/integration/controller_suite_test.go
@@ -43,12 +43,12 @@ import (
"sigs.k8s.io/controller-runtime/pkg/log/zap"
appsv1alpha1 "github.com/apecloud/kubeblocks/apis/apps/v1alpha1"
- dataprotectionv1alpha1 "github.com/apecloud/kubeblocks/apis/dataprotection/v1alpha1"
+ dpv1alpha1 "github.com/apecloud/kubeblocks/apis/dataprotection/v1alpha1"
"github.com/apecloud/kubeblocks/controllers/apps"
- "github.com/apecloud/kubeblocks/controllers/apps/components"
dpctrl "github.com/apecloud/kubeblocks/controllers/dataprotection"
"github.com/apecloud/kubeblocks/controllers/k8score"
cliutil "github.com/apecloud/kubeblocks/internal/cli/util"
+ "github.com/apecloud/kubeblocks/internal/common"
cfgcore "github.com/apecloud/kubeblocks/internal/configuration/core"
"github.com/apecloud/kubeblocks/internal/constant"
intctrlutil "github.com/apecloud/kubeblocks/internal/controllerutil"
@@ -103,7 +103,7 @@ func GetConsensusRoleCountMap(testCtx testutil.TestContext, k8sClient client.Cli
}
sts := stsList.Items[0]
- pods, err := components.GetPodListByStatefulSet(testCtx.Ctx, k8sClient, &sts)
+ pods, err := common.GetPodListByStatefulSet(testCtx.Ctx, k8sClient, &sts)
if err != nil {
return roleCountMap
@@ -256,7 +256,7 @@ var _ = BeforeSuite(func() {
err = appsv1alpha1.AddToScheme(scheme.Scheme)
Expect(err).NotTo(HaveOccurred())
- err = dataprotectionv1alpha1.AddToScheme(scheme.Scheme)
+ err = dpv1alpha1.AddToScheme(scheme.Scheme)
Expect(err).NotTo(HaveOccurred())
err = snapshotv1.AddToScheme(scheme.Scheme)
@@ -336,33 +336,33 @@ var _ = BeforeSuite(func() {
}).SetupWithManager(k8sManager)
Expect(err).ToNot(HaveOccurred())
- err = (&dpctrl.BackupPolicyReconciler{
+ err = (&dpctrl.BackupScheduleReconciler{
Client: k8sManager.GetClient(),
Scheme: k8sManager.GetScheme(),
Recorder: k8sManager.GetEventRecorderFor("backup-policy-controller"),
}).SetupWithManager(k8sManager)
Expect(err).ToNot(HaveOccurred())
- err = (&dpctrl.BackupToolReconciler{
+ err = (&dpctrl.ActionSetReconciler{
Client: k8sManager.GetClient(),
Scheme: k8sManager.GetScheme(),
Recorder: k8sManager.GetEventRecorderFor("backup-tool-controller"),
}).SetupWithManager(k8sManager)
Expect(err).ToNot(HaveOccurred())
- err = (&dpctrl.RestoreJobReconciler{
- Client: k8sManager.GetClient(),
- Scheme: k8sManager.GetScheme(),
- Recorder: k8sManager.GetEventRecorderFor("restore-job-controller"),
- }).SetupWithManager(k8sManager)
- Expect(err).ToNot(HaveOccurred())
-
- err = (&dpctrl.CronJobReconciler{
- Client: k8sManager.GetClient(),
- Scheme: k8sManager.GetScheme(),
- Recorder: k8sManager.GetEventRecorderFor("cronjob-controller"),
- }).SetupWithManager(k8sManager)
- Expect(err).ToNot(HaveOccurred())
+ // err = (&dpctrl.RestoreJobReconciler{
+ // Client: k8sManager.GetClient(),
+ // Scheme: k8sManager.GetScheme(),
+ // Recorder: k8sManager.GetEventRecorderFor("restore-job-controller"),
+ // }).SetupWithManager(k8sManager)
+ // Expect(err).ToNot(HaveOccurred())
+
+ // err = (&dpctrl.CronJobReconciler{
+ // Client: k8sManager.GetClient(),
+ // Scheme: k8sManager.GetScheme(),
+ // Recorder: k8sManager.GetEventRecorderFor("cronjob-controller"),
+ // }).SetupWithManager(k8sManager)
+ // Expect(err).ToNot(HaveOccurred())
// pulling docker images is slow
viper.SetDefault("EventuallyTimeout", time.Second*300)
diff --git a/test/integration/mysql_ha_test.go b/test/integration/mysql_ha_test.go
index 840d93d6673..75ad99c3ae5 100644
--- a/test/integration/mysql_ha_test.go
+++ b/test/integration/mysql_ha_test.go
@@ -29,7 +29,7 @@ import (
"sigs.k8s.io/controller-runtime/pkg/client"
appsv1alpha1 "github.com/apecloud/kubeblocks/apis/apps/v1alpha1"
- "github.com/apecloud/kubeblocks/controllers/apps/components"
+ "github.com/apecloud/kubeblocks/internal/common"
"github.com/apecloud/kubeblocks/internal/constant"
intctrlutil "github.com/apecloud/kubeblocks/internal/generics"
testapps "github.com/apecloud/kubeblocks/internal/testutil/apps"
@@ -121,7 +121,7 @@ var _ = Describe("MySQL High-Availability function", func() {
By("Checking pods' role label")
stsList := testk8s.ListAndCheckStatefulSet(&testCtx, clusterKey)
sts := &stsList.Items[0]
- pods, err := components.GetPodListByStatefulSet(ctx, k8sClient, sts)
+ pods, err := common.GetPodListByStatefulSet(ctx, k8sClient, sts)
Expect(err).To(Succeed())
// should have 3 pods
Expect(len(pods)).Should(Equal(3))
diff --git a/test/integration/mysql_reconfigure_test.go b/test/integration/mysql_reconfigure_test.go
index fb92347f641..0508fe4e1ea 100644
--- a/test/integration/mysql_reconfigure_test.go
+++ b/test/integration/mysql_reconfigure_test.go
@@ -27,9 +27,9 @@ import (
"sigs.k8s.io/controller-runtime/pkg/client"
appsv1alpha1 "github.com/apecloud/kubeblocks/apis/apps/v1alpha1"
- "github.com/apecloud/kubeblocks/controllers/apps/components"
clitypes "github.com/apecloud/kubeblocks/internal/cli/types"
cliutil "github.com/apecloud/kubeblocks/internal/cli/util"
+ "github.com/apecloud/kubeblocks/internal/common"
"github.com/apecloud/kubeblocks/internal/configuration/core"
"github.com/apecloud/kubeblocks/internal/generics"
testapps "github.com/apecloud/kubeblocks/internal/testutil/apps"
@@ -152,7 +152,7 @@ var _ = Describe("MySQL Reconfigure function", func() {
By("Checking pods' role label")
sts := testk8s.ListAndCheckStatefulSet(&testCtx, clusterKey).Items[0]
- pods, err := components.GetPodListByStatefulSet(testCtx.Ctx, k8sClient, &sts)
+ pods, err := common.GetPodListByStatefulSet(testCtx.Ctx, k8sClient, &sts)
Expect(err).To(Succeed())
Expect(len(pods)).Should(Equal(3))
diff --git a/test/integration/redis_hscale_test.go b/test/integration/redis_hscale_test.go
index 453d6002b49..4c5ddc4f9da 100644
--- a/test/integration/redis_hscale_test.go
+++ b/test/integration/redis_hscale_test.go
@@ -29,7 +29,7 @@ import (
"sigs.k8s.io/controller-runtime/pkg/client"
appsv1alpha1 "github.com/apecloud/kubeblocks/apis/apps/v1alpha1"
- "github.com/apecloud/kubeblocks/controllers/apps/components"
+ "github.com/apecloud/kubeblocks/internal/common"
"github.com/apecloud/kubeblocks/internal/constant"
intctrlutil "github.com/apecloud/kubeblocks/internal/generics"
testapps "github.com/apecloud/kubeblocks/internal/testutil/apps"
@@ -106,7 +106,7 @@ var _ = Describe("Redis Horizontal Scale function", func() {
Expect(len(stsList.Items)).Should(BeEquivalentTo(1))
By("Checking pods number and role label in StatefulSet")
- podList, err := components.GetPodListByStatefulSet(ctx, k8sClient, &stsList.Items[0])
+ podList, err := common.GetPodListByStatefulSet(ctx, k8sClient, &stsList.Items[0])
Expect(err).To(Succeed())
Expect(len(podList)).Should(BeEquivalentTo(replicas))
for _, pod := range podList {
diff --git a/test/testdata/backup/actionset.yaml b/test/testdata/backup/actionset.yaml
new file mode 100644
index 00000000000..3f9ec0602ca
--- /dev/null
+++ b/test/testdata/backup/actionset.yaml
@@ -0,0 +1,36 @@
+apiVersion: dataprotection.kubeblocks.io/v1alpha1
+kind: ActionSet
+metadata:
+ name: action-set-
+spec:
+ backupType: Full
+ env:
+ - name: DATA_DIR
+ value: /var/lib/mysql
+ backup:
+ backupData:
+ image: registry.cn-hangzhou.aliyuncs.com/apecloud/percona-xtrabackup
+ runOnTargetPodNode: true
+ command:
+ - sh
+ - -c
+ - echo "DB_HOST=${DP_DB_HOST} DB_USER=${DP_DB_USER} DB_PASSWORD=${DP_DB_PASSWORD} DATA_DIR=${DATA_DIR} BACKUP_DIR=${DP_BACKUP_DIR} BACKUP_NAME=${DP_BACKUP_NAME}";
+ mkdir -p /${BACKUP_DIR};
+ xtrabackup --compress --backup --safe-slave-backup --slave-info --stream=xbstream --host=${DP_DB_HOST} \
+ --user=${DP_DB_USER} --password=${DP_DB_PASSWORD} --datadir=${DATA_DIR} > /${DP_BACKUP_DIR}/${DP_BACKUP_NAME}.xbstream
+ restore:
+ prepareData:
+ image: registry.cn-hangzhou.aliyuncs.com/apecloud/percona-xtrabackup
+ command:
+ - sh
+ - -c
+ - |
+ echo "BACKUP_DIR=${DP_BACKUP_DIR} BACKUP_NAME=${DP_BACKUP_NAME} DATA_DIR=${DATA_DIR}" && \
+ mkdir -p /tmp/data/ && cd /tmp/data \
+ && xbstream -x < /${DP_BACKUP_DIR}/${DP_BACKUP_NAME}.xbstream \
+ && xtrabackup --decompress --target-dir=/tmp/data/ \
+ && find . -name "*.qp"|xargs rm -f \
+ && rm -rf ${DATA_DIR}/* \
+ && rsync -avrP /tmp/data/ ${DATA_DIR}/ \
+ && rm -rf /tmp/data/ \
+ && chmod -R 0777 ${DATA_DIR}
\ No newline at end of file
diff --git a/test/testdata/backup/backuptool.yaml b/test/testdata/backup/backuptool.yaml
deleted file mode 100644
index 4edaae0eda3..00000000000
--- a/test/testdata/backup/backuptool.yaml
+++ /dev/null
@@ -1,35 +0,0 @@
-apiVersion: dataprotection.kubeblocks.io/v1alpha1
-kind: BackupTool
-metadata:
- name: backup-tool-
-spec:
- image: registry.cn-hangzhou.aliyuncs.com/apecloud/percona-xtrabackup
- deployKind: job
- env:
- - name: DATA_DIR
- value: /var/lib/mysql
- physical:
- restoreCommands:
- - sh
- - -c
- - |
- echo "BACKUP_DIR=${BACKUP_DIR} BACKUP_NAME=${BACKUP_NAME} DATA_DIR=${DATA_DIR}" && \
- mkdir -p /tmp/data/ && cd /tmp/data \
- && xbstream -x < /${BACKUP_DIR}/${BACKUP_NAME}.xbstream \
- && xtrabackup --decompress --target-dir=/tmp/data/ \
- && find . -name "*.qp"|xargs rm -f \
- && rm -rf ${DATA_DIR}/* \
- && rsync -avrP /tmp/data/ ${DATA_DIR}/ \
- && rm -rf /tmp/data/ \
- && chmod -R 0777 ${DATA_DIR}
- incrementalRestoreCommands: []
- logical:
- restoreCommands: []
- incrementalRestoreCommands: []
- backupCommands:
- - sh
- - -c
- - echo "DB_HOST=${DB_HOST} DB_USER=${DB_USER} DB_PASSWORD=${DB_PASSWORD} DATA_DIR=${DATA_DIR} BACKUP_DIR=${BACKUP_DIR} BACKUP_NAME=${BACKUP_NAME}";
- mkdir -p /${BACKUP_DIR};
- xtrabackup --compress --backup --safe-slave-backup --slave-info --stream=xbstream --host=${DB_HOST} --user=${DB_USER} --password=${DB_PASSWORD} --datadir=${DATA_DIR} > /${BACKUP_DIR}/${BACKUP_NAME}.xbstream
- incrementalBackupCommands: []
\ No newline at end of file