From 372cf47380d4e6055f4d2e3be0b9019023b3cbff Mon Sep 17 00:00:00 2001 From: Wout Date: Thu, 9 Nov 2023 19:12:20 +0000 Subject: [PATCH 1/8] Added support for Discord webhooks --- bin/build | 4 ++-- helm/templates/deployment.yaml | 9 +++++++++ helm/templates/secret.yaml | 1 + helm/values.yaml | 9 +++++++-- shard.yml | 2 +- src/event.cr | 32 +++++++++++++++++++++++++++++++- 6 files changed, 51 insertions(+), 6 deletions(-) diff --git a/bin/build b/bin/build index 3f8ee4b..c1f8ed1 100755 --- a/bin/build +++ b/bin/build @@ -1,9 +1,9 @@ #!/bin/bash -docker run --rm -it --platform=linux/amd64 -v $(pwd):/workspace -w /workspace crystallang/crystal:latest-alpine crystal build src/velero-notifications.cr --static +docker run --rm -it --platform=linux/amd64 -v $(pwd):/workspace -w /workspace crystallang/crystal:latest-alpine /bin/sh -c "shards install && crystal build src/velero-notifications.cr --static" -IMAGE="vitobotta/velero-backup-notification" +IMAGE="woutthenines/velero-backup-notification" VERSION="$(git describe --tags --abbrev=0)" docker build --platform=linux -t ${IMAGE}:${VERSION} . diff --git a/helm/templates/deployment.yaml b/helm/templates/deployment.yaml index 5406618..0ed0398 100644 --- a/helm/templates/deployment.yaml +++ b/helm/templates/deployment.yaml @@ -52,6 +52,15 @@ spec: secretKeyRef: key: slack_channel name: velero-backup-notification-secrets + - name: ENABLE_DISCORD_NOTIFICATIONS + value: {{ .Values.discord.enabled | quote }} + - name: DISCORD_FAILURES_ONLY + value: {{ .Values.discord.failures_only | quote }} + - name: DISCORD_WEBHOOK + valueFrom: + secretKeyRef: + key: discord_webhook + name: velero-backup-notification-secrets - name: ENABLE_EMAIL_NOTIFICATIONS value: {{ .Values.email.enabled | quote }} - name: EMAIL_FAILURES_ONLY diff --git a/helm/templates/secret.yaml b/helm/templates/secret.yaml index de99291..65e0355 100644 --- a/helm/templates/secret.yaml +++ b/helm/templates/secret.yaml @@ -6,6 +6,7 @@ type: Opaque stringData: slack_webhook: {{ .Values.slack.webhook | quote }} slack_channel: {{ .Values.slack.channel | quote }} + discord_webhook: {{ .Values.discord.webhook | quote }} email_smtp_host: {{ .Values.email.smtp.host | quote }} email_smtp_port: {{ .Values.email.smtp.port | quote }} email_smtp_username: {{ .Values.email.smtp.username | quote }} diff --git a/helm/values.yaml b/helm/values.yaml index bd3f7d1..f8ac866 100644 --- a/helm/values.yaml +++ b/helm/values.yaml @@ -1,6 +1,6 @@ image: - repository: vitobotta/velero-backup-notification - tag: v1.0.0 + repository: woutthenines/velero-notifications + tag: v0.0.1 slack: enabled: false @@ -9,6 +9,11 @@ slack: channel: "stuff" username: Velero +discord: + enabled: false + failures_only: true + webhook: "https://...." + email: enabled: false failures_only: true diff --git a/shard.yml b/shard.yml index bc3c329..bfcc334 100644 --- a/shard.yml +++ b/shard.yml @@ -15,7 +15,7 @@ license: MIT dependencies: kube-client: github: spoved/kube-client.cr - branch: kalinon/issue12 + version: 0.4.8 retriable: git: https://github.com/sija/retriable.cr.git version: 0.2.4 diff --git a/src/event.cr b/src/event.cr index cc98604..82d5a53 100644 --- a/src/event.cr +++ b/src/event.cr @@ -4,6 +4,7 @@ require "email" class Event SLACK_WEBHOOK = ENV.fetch("SLACK_WEBHOOK", "") + DISCORD_WEBHOOK = ENV.fetch("DISCORD_WEBHOOK", "") EMAIL_SMTP_HOST = ENV.fetch("EMAIL_SMTP_HOST", "") EMAIL_SMTP_PORT = ENV.fetch("EMAIL_SMTP_PORT", "") EMAIL_SMTP_USERNAME = ENV.fetch("EMAIL_SMTP_USERNAME", "") @@ -25,6 +26,7 @@ class Event Log.info { notification_subject } send_slack_notification if send_slack_notification? + send_discord_notification if send_discord_notification? send_email_notification if send_email_notification? send_webhook_notification if send_webhook_notification? end @@ -83,6 +85,35 @@ class Event ) end + def send_discord_notification? + send_notification?(:discord) + end + + # Add a method to send notifications to Discord + def send_discord_notification + if DISCORD_WEBHOOK.blank? + Log.info { "Ensure the DISCORD_WEBHOOK environment variable is set" } + raise Exception.new("Discord configuration missing") + end + + color = phase == "Completed" ? 0x36a64f : 0xa30202 + payload = { + "embeds" => [ + { + "title" => notification_subject, + "description" => notification_body, + "color" => color + } + ] + }.to_json + + HTTP::Client.post( + DISCORD_WEBHOOK, + headers: HTTP::Headers{"Content-type" => "application/json"}, + body: payload + ) + end + private def email_client : EMail::Client @email_client ||= begin if EMAIL_SMTP_HOST.blank? || EMAIL_SMTP_PORT.blank? || EMAIL_SMTP_USERNAME.blank? || EMAIL_SMTP_PASSWORD.blank? || EMAIL_FROM_ADDRESS.blank? || EMAIL_TO_ADDRESS.blank? @@ -138,4 +169,3 @@ class Event HTTP::Client.get(WEBHOOK_URL) end end - From 419171d4fabf2a41303085a6e65880d09ef6b5af Mon Sep 17 00:00:00 2001 From: Wout Date: Thu, 9 Nov 2023 20:58:28 +0000 Subject: [PATCH 2/8] Updated version --- helm/Chart.yaml | 4 ++-- helm/values.yaml | 4 ++-- shard.lock | 6 +++--- shard.yml | 2 +- src/velero-notifications.cr | 2 +- 5 files changed, 9 insertions(+), 9 deletions(-) diff --git a/helm/Chart.yaml b/helm/Chart.yaml index c4aa573..f94de16 100644 --- a/helm/Chart.yaml +++ b/helm/Chart.yaml @@ -1,5 +1,5 @@ apiVersion: v1 -appVersion: "1.0" +appVersion: "v1.0.1" description: A Helm chart to send email/Slack notifications for Velero backups/restores name: velero-backup-notification -version: 0.1.0 +version: 1.0.1 diff --git a/helm/values.yaml b/helm/values.yaml index f8ac866..e24a8fa 100644 --- a/helm/values.yaml +++ b/helm/values.yaml @@ -1,6 +1,6 @@ image: - repository: woutthenines/velero-notifications - tag: v0.0.1 + repository: woutthenines/velero-backup-notifications + tag: v1.0.1 slack: enabled: false diff --git a/shard.lock b/shard.lock index 72b7873..2eac0ff 100644 --- a/shard.lock +++ b/shard.lock @@ -2,7 +2,7 @@ version: 2.0 shards: db: git: https://github.com/crystal-lang/crystal-db.git - version: 0.11.0 + version: 0.12.0 email: git: https://github.com/arcage/crystal-email.git @@ -14,11 +14,11 @@ shards: k8s: git: https://github.com/spoved/k8s.cr.git - version: 0.1.11 + version: 0.1.12 kube-client: git: https://github.com/spoved/kube-client.cr.git - version: 0.4.6+git.commit.8778313a458f239376c7f05896a79fd8414e4139 + version: 0.4.8 retriable: git: https://github.com/sija/retriable.cr.git diff --git a/shard.yml b/shard.yml index bfcc334..23c23ad 100644 --- a/shard.yml +++ b/shard.yml @@ -1,5 +1,5 @@ name: velero-notifications -version: 0.1.0 +version: 1.0.1 authors: - Vito Botta diff --git a/src/velero-notifications.cr b/src/velero-notifications.cr index 6c43a05..ad92f63 100644 --- a/src/velero-notifications.cr +++ b/src/velero-notifications.cr @@ -1,7 +1,7 @@ require "./controller" module Velero::Notifications - VERSION = "1.0.0" + VERSION = "1.0.1" end From b94c968ac173852062114d9f539f3688fbd267de Mon Sep 17 00:00:00 2001 From: Wout Date: Thu, 9 Nov 2023 21:03:55 +0000 Subject: [PATCH 3/8] Update README.md --- README.md | 5 ++++- 1 file changed, 4 insertions(+), 1 deletion(-) diff --git a/README.md b/README.md index d9c1fc7..c44f4ee 100644 --- a/README.md +++ b/README.md @@ -14,7 +14,7 @@ # velero-notifications -This is a simple Kubernetes controller written in Crystal that sends Email/Slack/webhook notifications when backups are performed by [Velero](https://velero.io/) in a [Kubernetes](https://kubernetes.io/) cluster. +This is a simple Kubernetes controller written in Crystal that sends Email/Slack/Discord/webhook notifications when backups are performed by [Velero](https://velero.io/) in a [Kubernetes](https://kubernetes.io/) cluster. ![Screenshot](slack.png?raw=true "Screenshot") @@ -35,6 +35,9 @@ helm upgrade --install \ --set slack.webhook=https://... \ --set slack.channel=velero \ --set slack.username=Velero \ + --set discord.enabled=true \ + --set discord.failures_only=false \ + --set discord.webhook=https://... \ --set email.enabled=true \ --set email.failures_only=true \ --set email.smtp.host=... \ From f6bf4de2c957edbbcf131d8195caf9382b4ff334 Mon Sep 17 00:00:00 2001 From: Wout Date: Thu, 9 Nov 2023 21:09:28 +0000 Subject: [PATCH 4/8] Fix typo --- helm/values.yaml | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/helm/values.yaml b/helm/values.yaml index e24a8fa..934228b 100644 --- a/helm/values.yaml +++ b/helm/values.yaml @@ -1,5 +1,5 @@ image: - repository: woutthenines/velero-backup-notifications + repository: woutthenines/velero-backup-notification tag: v1.0.1 slack: From 99e4a2b5954829ffc05e6af2cd56b32468d3bf31 Mon Sep 17 00:00:00 2001 From: Wout Date: Thu, 9 Nov 2023 23:10:50 +0000 Subject: [PATCH 5/8] Added discord mentions --- README.md | 3 +++ helm/Chart.yaml | 4 ++-- helm/templates/deployment.yaml | 6 ++++++ helm/values.yaml | 6 +++++- shard.yml | 2 +- src/event.cr | 15 +++++++++++++++ src/velero-notifications.cr | 2 +- 7 files changed, 33 insertions(+), 5 deletions(-) diff --git a/README.md b/README.md index c44f4ee..ffe64d0 100644 --- a/README.md +++ b/README.md @@ -38,6 +38,9 @@ helm upgrade --install \ --set discord.enabled=true \ --set discord.failures_only=false \ --set discord.webhook=https://... \ + --set discord.mentions.enabled=false \ + --set discord.mentions.failures_only=true \ + --set discord.mentions.role_id="1234567890" \ --set email.enabled=true \ --set email.failures_only=true \ --set email.smtp.host=... \ diff --git a/helm/Chart.yaml b/helm/Chart.yaml index f94de16..73ae40b 100644 --- a/helm/Chart.yaml +++ b/helm/Chart.yaml @@ -1,5 +1,5 @@ apiVersion: v1 -appVersion: "v1.0.1" +appVersion: "v1.0.2" description: A Helm chart to send email/Slack notifications for Velero backups/restores name: velero-backup-notification -version: 1.0.1 +version: 1.0.2 diff --git a/helm/templates/deployment.yaml b/helm/templates/deployment.yaml index 0ed0398..acc801a 100644 --- a/helm/templates/deployment.yaml +++ b/helm/templates/deployment.yaml @@ -61,6 +61,12 @@ spec: secretKeyRef: key: discord_webhook name: velero-backup-notification-secrets + - name: ENABLE_DISCORD_MENTIONS + value: {{ .Values.discord.mentions.enabled | quote }} + - name: DISCORD_MENTIONS_FAILURES_ONLY + value: {{ .Values.discord.mentions.failures_only | quote }} + - name: DISCORD_MENTIONS_ROLE_ID + value: {{ .Values.discord.mentions.role_id | quote }} - name: ENABLE_EMAIL_NOTIFICATIONS value: {{ .Values.email.enabled | quote }} - name: EMAIL_FAILURES_ONLY diff --git a/helm/values.yaml b/helm/values.yaml index 934228b..fc22f83 100644 --- a/helm/values.yaml +++ b/helm/values.yaml @@ -1,6 +1,6 @@ image: repository: woutthenines/velero-backup-notification - tag: v1.0.1 + tag: v1.0.2 slack: enabled: false @@ -13,6 +13,10 @@ discord: enabled: false failures_only: true webhook: "https://...." + mentions: + enabled: false + failures_only: true + role_id: "1234567890" email: enabled: false diff --git a/shard.yml b/shard.yml index 23c23ad..c2205a7 100644 --- a/shard.yml +++ b/shard.yml @@ -1,5 +1,5 @@ name: velero-notifications -version: 1.0.1 +version: 1.0.2 authors: - Vito Botta diff --git a/src/event.cr b/src/event.cr index 82d5a53..c2d2c12 100644 --- a/src/event.cr +++ b/src/event.cr @@ -5,6 +5,9 @@ require "email" class Event SLACK_WEBHOOK = ENV.fetch("SLACK_WEBHOOK", "") DISCORD_WEBHOOK = ENV.fetch("DISCORD_WEBHOOK", "") + ENABLE_DISCORD_MENTIONS = ENV.fetch("ENABLE_DISCORD_MENTIONS", "false").downcase + DISCORD_MENTIONS_FAILURES_ONLY = ENV.fetch("DISCORD_MENTIONS_FAILURES_ONLY", "false").downcase + DISCORD_MENTIONS_ROLE_ID = ENV.fetch("DISCORD_MENTIONS_ROLE_ID", "") EMAIL_SMTP_HOST = ENV.fetch("EMAIL_SMTP_HOST", "") EMAIL_SMTP_PORT = ENV.fetch("EMAIL_SMTP_PORT", "") EMAIL_SMTP_USERNAME = ENV.fetch("EMAIL_SMTP_USERNAME", "") @@ -96,6 +99,18 @@ class Event raise Exception.new("Discord configuration missing") end + if ENABLE_DISCORD_MENTIONS == "true" + if DISCORD_MENTIONS_ROLE_ID.blank? + Log.info { "Ensure the DISCORD_MENTIONS_ROLE_ID environment variable is set" } + raise Exception.new("Discord mentions configuration missing") + end + + failures_only = DISCORD_MENTIONS_FAILURES_ONLY == "true" + succeeded = phase == "Completed" + + notification_body = !failures_only || succeeded ? "#{notification_body} <@&#{DISCORD_MENTIONS_ROLE_ID}>" : notification_body + end + color = phase == "Completed" ? 0x36a64f : 0xa30202 payload = { "embeds" => [ diff --git a/src/velero-notifications.cr b/src/velero-notifications.cr index ad92f63..d13ee58 100644 --- a/src/velero-notifications.cr +++ b/src/velero-notifications.cr @@ -1,7 +1,7 @@ require "./controller" module Velero::Notifications - VERSION = "1.0.1" + VERSION = "1.0.2" end From ea943d746393a821eb9f66313627a607da31552a Mon Sep 17 00:00:00 2001 From: Wout Date: Fri, 10 Nov 2023 12:15:06 +0000 Subject: [PATCH 6/8] Discord mention fix --- README.md | 2 ++ discord.png | Bin 0 -> 19297 bytes src/event.cr | 3 ++- 3 files changed, 4 insertions(+), 1 deletion(-) create mode 100644 discord.png diff --git a/README.md b/README.md index ffe64d0..737f3d8 100644 --- a/README.md +++ b/README.md @@ -18,6 +18,8 @@ This is a simple Kubernetes controller written in Crystal that sends Email/Slack ![Screenshot](slack.png?raw=true "Screenshot") +![Screenshot](discord.png?raw=true "Screenshot") + If you like this or any of my other projects and would like to help with their development, consider [becoming a sponsor](https://github.com/sponsors/vitobotta). ## Installation diff --git a/discord.png b/discord.png new file mode 100644 index 0000000000000000000000000000000000000000..5f3594d0d78867058b6f64b0f9deab46fb602954 GIT binary patch literal 19297 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 literal 0 HcmV?d00001 diff --git a/src/event.cr b/src/event.cr index c2d2c12..c25ea94 100644 --- a/src/event.cr +++ b/src/event.cr @@ -108,11 +108,12 @@ class Event failures_only = DISCORD_MENTIONS_FAILURES_ONLY == "true" succeeded = phase == "Completed" - notification_body = !failures_only || succeeded ? "#{notification_body} <@&#{DISCORD_MENTIONS_ROLE_ID}>" : notification_body + notification_mention = !failures_only || (failures_only && !succeeded) ? "<@&#{DISCORD_MENTIONS_ROLE_ID}>" : nil end color = phase == "Completed" ? 0x36a64f : 0xa30202 payload = { + "content" => notification_mention.nil? ? "" : notification_mention, "embeds" => [ { "title" => notification_subject, From a8ad5249be39bef44595cf99aec88811661b422c Mon Sep 17 00:00:00 2001 From: Wout Date: Tue, 5 Dec 2023 17:41:35 +0000 Subject: [PATCH 7/8] Update to Kubernetes v1.28 --- helm/Chart.yaml | 4 ++-- helm/values.yaml | 2 +- shard.yml | 2 +- src/controller.cr | 2 +- src/velero-notifications.cr | 2 +- 5 files changed, 6 insertions(+), 6 deletions(-) diff --git a/helm/Chart.yaml b/helm/Chart.yaml index 73ae40b..5b67cdb 100644 --- a/helm/Chart.yaml +++ b/helm/Chart.yaml @@ -1,5 +1,5 @@ apiVersion: v1 -appVersion: "v1.0.2" +appVersion: "v1.0.3" description: A Helm chart to send email/Slack notifications for Velero backups/restores name: velero-backup-notification -version: 1.0.2 +version: 1.0.3 diff --git a/helm/values.yaml b/helm/values.yaml index fc22f83..d11f164 100644 --- a/helm/values.yaml +++ b/helm/values.yaml @@ -1,6 +1,6 @@ image: repository: woutthenines/velero-backup-notification - tag: v1.0.2 + tag: v1.0.3 slack: enabled: false diff --git a/shard.yml b/shard.yml index c2205a7..cf21c93 100644 --- a/shard.yml +++ b/shard.yml @@ -1,5 +1,5 @@ name: velero-notifications -version: 1.0.2 +version: 1.0.3 authors: - Vito Botta diff --git a/src/controller.cr b/src/controller.cr index 851fbd6..aa9749f 100644 --- a/src/controller.cr +++ b/src/controller.cr @@ -1,5 +1,5 @@ require "log" -require "kube-client/v1.26" +require "kube-client/v1.28" require "retriable" require "./crds/velero/v1/backup_spec" require "./crds/velero/v1/backup_status" diff --git a/src/velero-notifications.cr b/src/velero-notifications.cr index d13ee58..fd018b0 100644 --- a/src/velero-notifications.cr +++ b/src/velero-notifications.cr @@ -1,7 +1,7 @@ require "./controller" module Velero::Notifications - VERSION = "1.0.2" + VERSION = "1.0.3" end From ad78cbd9b142aa5c4792d9f32e934340f52e3f5b Mon Sep 17 00:00:00 2001 From: Wout Date: Thu, 28 Dec 2023 14:26:11 +0000 Subject: [PATCH 8/8] Updated Velero CRDs --- bin/build | 2 +- crds.yaml | 73 +++++++++++++++++------------ src/crds/velero/v1/backup_spec.cr | 3 ++ src/crds/velero/v1/backup_status.cr | 1 + 4 files changed, 47 insertions(+), 32 deletions(-) diff --git a/bin/build b/bin/build index c1f8ed1..f17d5bf 100755 --- a/bin/build +++ b/bin/build @@ -1,6 +1,6 @@ #!/bin/bash -docker run --rm -it --platform=linux/amd64 -v $(pwd):/workspace -w /workspace crystallang/crystal:latest-alpine /bin/sh -c "shards install && crystal build src/velero-notifications.cr --static" +docker run --rm -it --platform=linux/amd64 -v $(pwd):/workspace -w /workspace crystallang/crystal:latest-alpine /bin/sh -c "shards install && crystal run ./lib/k8s/bin/gen_crd.cr -- ./crds.yaml ./src/crds && crystal build src/velero-notifications.cr --static" IMAGE="woutthenines/velero-backup-notification" diff --git a/crds.yaml b/crds.yaml index 07496b5..c697675 100644 --- a/crds.yaml +++ b/crds.yaml @@ -1,20 +1,11 @@ +--- apiVersion: apiextensions.k8s.io/v1 kind: CustomResourceDefinition metadata: annotations: - controller-gen.kubebuilder.io/version: v0.7.0 - kubectl.kubernetes.io/last-applied-configuration: | - {"apiVersion":"apiextensions.k8s.io/v1","kind":"CustomResourceDefinition","metadata":{"annotations":{"controller-gen.kubebuilder.io/version":"v0.7.0"},"creationTimestamp":null,"labels":{"component":"velero"},"name":"backups.velero.io"},"spec":{"group":"velero.io","names":{"kind":"Backup","listKind":"BackupList","plural":"backups","singular":"backup"},"scope":"Namespaced","versions":[{"name":"v1","schema":{"openAPIV3Schema":{"description":"Backup is a Velero resource that represents the capture of Kubernetes cluster state at a point in time (API objects and associated volume state).","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":"BackupSpec defines the specification for a Velero backup.","properties":{"csiSnapshotTimeout":{"description":"CSISnapshotTimeout specifies the time used to wait for CSI VolumeSnapshot status turns to ReadyToUse during creation, before returning error as timeout. The default value is 10 minute.","type":"string"},"defaultVolumesToFsBackup":{"description":"DefaultVolumesToFsBackup specifies whether pod volume file system backup should be used for all volumes by default.","nullable":true,"type":"boolean"},"defaultVolumesToRestic":{"description":"DefaultVolumesToRestic specifies whether restic should be used to take a backup of all pod volumes by default. \n Deprecated: this field is no longer used and will be removed entirely in future. Use DefaultVolumesToFsBackup instead.","nullable":true,"type":"boolean"},"excludedClusterScopedResources":{"description":"ExcludedClusterScopedResources is a slice of cluster-scoped resource type names to exclude from the backup. If set to \"*\", all cluster-scoped resource types are excluded. The default value is empty.","items":{"type":"string"},"nullable":true,"type":"array"},"excludedNamespaceScopedResources":{"description":"ExcludedNamespaceScopedResources is a slice of namespace-scoped resource type names to exclude from the backup. If set to \"*\", all namespace-scoped resource types are excluded. The default value is empty.","items":{"type":"string"},"nullable":true,"type":"array"},"excludedNamespaces":{"description":"ExcludedNamespaces contains a list of namespaces that are not included in the backup.","items":{"type":"string"},"nullable":true,"type":"array"},"excludedResources":{"description":"ExcludedResources is a slice of resource names that are not included in the backup.","items":{"type":"string"},"nullable":true,"type":"array"},"hooks":{"description":"Hooks represent custom behaviors that should be executed at different phases of the backup.","properties":{"resources":{"description":"Resources are hooks that should be executed when backing up individual instances of a resource.","items":{"description":"BackupResourceHookSpec defines one or more BackupResourceHooks that should be executed based on the rules defined for namespaces, resources, and label selector.","properties":{"excludedNamespaces":{"description":"ExcludedNamespaces specifies the namespaces to which this hook spec does not apply.","items":{"type":"string"},"nullable":true,"type":"array"},"excludedResources":{"description":"ExcludedResources specifies the resources to which this hook spec does not apply.","items":{"type":"string"},"nullable":true,"type":"array"},"includedNamespaces":{"description":"IncludedNamespaces specifies the namespaces to which this hook spec applies. If empty, it applies to all namespaces.","items":{"type":"string"},"nullable":true,"type":"array"},"includedResources":{"description":"IncludedResources specifies the resources to which this hook spec applies. If empty, it applies to all resources.","items":{"type":"string"},"nullable":true,"type":"array"},"labelSelector":{"description":"LabelSelector, if specified, filters the resources to which this hook spec applies.","nullable":true,"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"},"name":{"description":"Name is the name of this hook.","type":"string"},"post":{"description":"PostHooks is a list of BackupResourceHooks to execute after storing the item in the backup. These are executed after all \"additional items\" from item actions are processed.","items":{"description":"BackupResourceHook defines a hook for a resource.","properties":{"exec":{"description":"Exec defines an exec hook.","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":{"description":"OnError specifies how Velero should behave if it encounters an error executing this hook.","enum":["Continue","Fail"],"type":"string"},"timeout":{"description":"Timeout defines the maximum amount of time Velero should wait for the hook to complete before considering the execution a failure.","type":"string"}},"required":["command"],"type":"object"}},"required":["exec"],"type":"object"},"type":"array"},"pre":{"description":"PreHooks is a list of BackupResourceHooks to execute prior to storing the item in the backup. These are executed before any \"additional items\" from item actions are processed.","items":{"description":"BackupResourceHook defines a hook for a resource.","properties":{"exec":{"description":"Exec defines an exec hook.","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":{"description":"OnError specifies how Velero should behave if it encounters an error executing this hook.","enum":["Continue","Fail"],"type":"string"},"timeout":{"description":"Timeout defines the maximum amount of time Velero should wait for the hook to complete before considering the execution a failure.","type":"string"}},"required":["command"],"type":"object"}},"required":["exec"],"type":"object"},"type":"array"}},"required":["name"],"type":"object"},"nullable":true,"type":"array"}},"type":"object"},"includeClusterResources":{"description":"IncludeClusterResources specifies whether cluster-scoped resources should be included for consideration in the backup.","nullable":true,"type":"boolean"},"includedClusterScopedResources":{"description":"IncludedClusterScopedResources is a slice of cluster-scoped resource type names to include in the backup. If set to \"*\", all cluster-scoped resource types are included. The default value is empty, which means only related cluster-scoped resources are included.","items":{"type":"string"},"nullable":true,"type":"array"},"includedNamespaceScopedResources":{"description":"IncludedNamespaceScopedResources is a slice of namespace-scoped resource type names to include in the backup. The default value is \"*\".","items":{"type":"string"},"nullable":true,"type":"array"},"includedNamespaces":{"description":"IncludedNamespaces is a slice of namespace names to include objects from. If empty, all namespaces are included.","items":{"type":"string"},"nullable":true,"type":"array"},"includedResources":{"description":"IncludedResources is a slice of resource names to include in the backup. If empty, all resources are included.","items":{"type":"string"},"nullable":true,"type":"array"},"itemOperationTimeout":{"description":"ItemOperationTimeout specifies the time used to wait for asynchronous BackupItemAction operations The default value is 1 hour.","type":"string"},"labelSelector":{"description":"LabelSelector is a metav1.LabelSelector to filter with when adding individual objects to the backup. If empty or nil, all objects are included. Optional.","nullable":true,"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"},"metadata":{"properties":{"labels":{"additionalProperties":{"type":"string"},"type":"object"}},"type":"object"},"orLabelSelectors":{"description":"OrLabelSelectors is list of metav1.LabelSelector to filter with when adding individual objects to the backup. If multiple provided they will be joined by the OR operator. LabelSelector as well as OrLabelSelectors cannot co-exist in backup request, only one of them can be used.","items":{"description":"A label selector is a label query over a set of resources. The result of matchLabels and matchExpressions are ANDed. An empty label selector matches all objects. A null label selector matches no objects.","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"},"nullable":true,"type":"array"},"orderedResources":{"additionalProperties":{"type":"string"},"description":"OrderedResources specifies the backup order of resources of specific Kind. The map key is the resource name and value is a list of object names separated by commas. Each resource name has format \"namespace/objectname\". For cluster resources, simply use \"objectname\".","nullable":true,"type":"object"},"resourcePolicy":{"description":"ResourcePolicy specifies the referenced resource policies that backup should follow","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"},"snapshotVolumes":{"description":"SnapshotVolumes specifies whether to take snapshots of any PV's referenced in the set of objects included in the Backup.","nullable":true,"type":"boolean"},"storageLocation":{"description":"StorageLocation is a string containing the name of a BackupStorageLocation where the backup should be stored.","type":"string"},"ttl":{"description":"TTL is a time.Duration-parseable string describing how long the Backup should be retained for.","type":"string"},"volumeSnapshotLocations":{"description":"VolumeSnapshotLocations is a list containing names of VolumeSnapshotLocations associated with this backup.","items":{"type":"string"},"type":"array"}},"type":"object"},"status":{"description":"BackupStatus captures the current status of a Velero backup.","properties":{"backupItemOperationsAttempted":{"description":"BackupItemOperationsAttempted is the total number of attempted async BackupItemAction operations for this backup.","type":"integer"},"backupItemOperationsCompleted":{"description":"BackupItemOperationsCompleted is the total number of successfully completed async BackupItemAction operations for this backup.","type":"integer"},"backupItemOperationsFailed":{"description":"BackupItemOperationsFailed is the total number of async BackupItemAction operations for this backup which ended with an error.","type":"integer"},"completionTimestamp":{"description":"CompletionTimestamp records the time a backup was completed. Completion time is recorded even on failed backups. Completion time is recorded before uploading the backup object. The server's time is used for CompletionTimestamps","format":"date-time","nullable":true,"type":"string"},"csiVolumeSnapshotsAttempted":{"description":"CSIVolumeSnapshotsAttempted is the total number of attempted CSI VolumeSnapshots for this backup.","type":"integer"},"csiVolumeSnapshotsCompleted":{"description":"CSIVolumeSnapshotsCompleted is the total number of successfully completed CSI VolumeSnapshots for this backup.","type":"integer"},"errors":{"description":"Errors is a count of all error messages that were generated during execution of the backup. The actual errors are in the backup's log file in object storage.","type":"integer"},"expiration":{"description":"Expiration is when this Backup is eligible for garbage-collection.","format":"date-time","nullable":true,"type":"string"},"failureReason":{"description":"FailureReason is an error that caused the entire backup to fail.","type":"string"},"formatVersion":{"description":"FormatVersion is the backup format version, including major, minor, and patch version.","type":"string"},"phase":{"description":"Phase is the current state of the Backup.","enum":["New","FailedValidation","InProgress","WaitingForPluginOperations","WaitingForPluginOperationsPartiallyFailed","Finalizing","FinalizingPartiallyFailed","Completed","PartiallyFailed","Failed","Deleting"],"type":"string"},"progress":{"description":"Progress contains information about the backup's execution progress. Note that this information is best-effort only -- if Velero fails to update it during a backup for any reason, it may be inaccurate/stale.","nullable":true,"properties":{"itemsBackedUp":{"description":"ItemsBackedUp is the number of items that have actually been written to the backup tarball so far.","type":"integer"},"totalItems":{"description":"TotalItems is the total number of items to be backed up. This number may change throughout the execution of the backup due to plugins that return additional related items to back up, the velero.io/exclude-from-backup label, and various other filters that happen as items are processed.","type":"integer"}},"type":"object"},"startTimestamp":{"description":"StartTimestamp records the time a backup was started. Separate from CreationTimestamp, since that value changes on restores. The server's time is used for StartTimestamps","format":"date-time","nullable":true,"type":"string"},"validationErrors":{"description":"ValidationErrors is a slice of all validation errors (if applicable).","items":{"type":"string"},"nullable":true,"type":"array"},"version":{"description":"Version is the backup format major version. Deprecated: Please see FormatVersion","type":"integer"},"volumeSnapshotsAttempted":{"description":"VolumeSnapshotsAttempted is the total number of attempted volume snapshots for this backup.","type":"integer"},"volumeSnapshotsCompleted":{"description":"VolumeSnapshotsCompleted is the total number of successfully completed volume snapshots for this backup.","type":"integer"},"warnings":{"description":"Warnings is a count of all warning messages that were generated during execution of the backup. The actual warnings are in the backup's log file in object storage.","type":"integer"}},"type":"object"}},"type":"object"}},"served":true,"storage":true}]}} - creationTimestamp: "2023-04-16T10:51:33Z" - generation: 2 - labels: - component: velero + controller-gen.kubebuilder.io/version: v0.12.0 name: backups.velero.io - resourceVersion: "6920040" - uid: b489ac82-cf5b-487f-b471-8b8e985d61b7 spec: - conversion: - strategy: None group: velero.io names: kind: Backup @@ -49,6 +40,11 @@ spec: CSI VolumeSnapshot status turns to ReadyToUse during creation, before returning error as timeout. The default value is 10 minute. type: string + datamover: + description: DataMover specifies the data mover to be used by the + backup. If DataMover is "" or "velero", the built-in data mover + will be used. + type: string defaultVolumesToFsBackup: description: DefaultVolumesToFsBackup specifies whether pod volume file system backup should be used for all volumes by default. @@ -182,6 +178,7 @@ spec: contains only "value". The requirements are ANDed. type: object type: object + x-kubernetes-map-type: atomic name: description: Name is the name of this hook. type: string @@ -367,6 +364,7 @@ spec: are ANDed. type: object type: object + x-kubernetes-map-type: atomic metadata: properties: labels: @@ -427,6 +425,7 @@ spec: are ANDed. type: object type: object + x-kubernetes-map-type: atomic nullable: true type: array orderedResources: @@ -459,6 +458,12 @@ spec: - kind - name type: object + x-kubernetes-map-type: atomic + snapshotMoveData: + description: SnapshotMoveData specifies whether snapshot data should + be moved + nullable: true + type: boolean snapshotVolumes: description: SnapshotVolumes specifies whether to take snapshots of any PV's referenced in the set of objects included in the Backup. @@ -472,6 +477,15 @@ spec: description: TTL is a time.Duration-parseable string describing how long the Backup should be retained for. type: string + uploaderConfig: + description: UploaderConfig specifies the configuration for the uploader. + nullable: true + properties: + parallelFilesUpload: + description: ParallelFilesUpload is the number of files parallel + uploads to perform when using the uploader. + type: integer + type: object volumeSnapshotLocations: description: VolumeSnapshotLocations is a list containing names of VolumeSnapshotLocations associated with this backup. @@ -530,6 +544,22 @@ spec: description: FormatVersion is the backup format version, including major, minor, and patch version. type: string + hookStatus: + description: HookStatus contains information about the status of the + hooks. + nullable: true + properties: + hooksAttempted: + description: HooksAttempted is the total number of attempted hooks + Specifically, HooksAttempted represents the number of hooks + that failed to execute and the number of hooks that executed + successfully. + type: integer + hooksFailed: + description: HooksFailed is the total number of hooks which ended + with an error + type: integer + type: object phase: description: Phase is the current state of the Backup. enum: @@ -597,23 +627,4 @@ spec: type: object type: object served: true - storage: true -status: - acceptedNames: - kind: Backup - listKind: BackupList - plural: backups - singular: backup - conditions: - - lastTransitionTime: "2023-04-16T10:51:33Z" - message: no conflicts found - reason: NoConflicts - status: "True" - type: NamesAccepted - - lastTransitionTime: "2023-04-16T10:51:33Z" - message: the initial names have been accepted - reason: InitialNamesAccepted - status: "True" - type: Established - storedVersions: - - v1 + storage: true \ No newline at end of file diff --git a/src/crds/velero/v1/backup_spec.cr b/src/crds/velero/v1/backup_spec.cr index fddf720..98f8f75 100644 --- a/src/crds/velero/v1/backup_spec.cr +++ b/src/crds/velero/v1/backup_spec.cr @@ -8,6 +8,7 @@ require "json" properties: [ {name: "csi_snapshot_timeout", kind: String, key: "csiSnapshotTimeout", nilable: true, read_only: false, description: "CSISnapshotTimeout specifies the time used to wait for CSI VolumeSnapshot status turns to ReadyToUse during creation, before returning error as timeout. The default value is 10 minute."}, + {name: "datamover", kind: String, key: "datamover", nilable: true, read_only: false, description: "DataMover specifies the data mover to be used by the backup. If DataMover is \"\" or \"velero\", the built-in data mover will be used."}, {name: "default_volumes_to_fs_backup", kind: ::Bool, key: "defaultVolumesToFsBackup", nilable: true, read_only: false, description: "DefaultVolumesToFsBackup specifies whether pod volume file system backup should be used for all volumes by default."}, {name: "default_volumes_to_restic", kind: ::Bool, key: "defaultVolumesToRestic", nilable: true, read_only: false, description: "DefaultVolumesToRestic specifies whether restic should be used to take a backup of all pod volumes by default. \n Deprecated: this field is no longer used and will be removed entirely in future. Use DefaultVolumesToFsBackup instead."}, {name: "excluded_cluster_scoped_resources", kind: ::Array(String), key: "excludedClusterScopedResources", nilable: true, read_only: false, description: "ExcludedClusterScopedResources is a slice of cluster-scoped resource type names to exclude from the backup. If set to \"*\", all cluster-scoped resource types are excluded. The default value is empty."}, @@ -26,9 +27,11 @@ require "json" {name: "or_label_selectors", kind: Union(::Array(::Hash(String, ::Array(::Hash(String, String | ::Array(String))) | ::Hash(String, String)))), key: "orLabelSelectors", nilable: true, read_only: false, description: "OrLabelSelectors is list of metav1.LabelSelector to filter with when adding individual objects to the backup. If multiple provided they will be joined by the OR operator. LabelSelector as well as OrLabelSelectors cannot co-exist in backup request, only one of them can be used."}, {name: "ordered_resources", kind: ::Hash(String, String), key: "orderedResources", nilable: true, read_only: false, description: "OrderedResources specifies the backup order of resources of specific Kind. The map key is the resource name and value is a list of object names separated by commas. Each resource name has format [\"namespace/objectname\". For cluster resources, simply use \"objectname\".](\"namespace/objectname\". For cluster resources, simply use \"objectname\".)"}, {name: "resource_policy", kind: ::Hash(String, String), key: "resourcePolicy", nilable: true, read_only: false, description: "ResourcePolicy specifies the referenced resource policies that backup should follow"}, + {name: "snapshot_move_data", kind: ::Bool, key: "snapshotMoveData", nilable: true, read_only: false, description: "SnapshotMoveData specifies whether snapshot data should be moved"}, {name: "snapshot_volumes", kind: ::Bool, key: "snapshotVolumes", nilable: true, read_only: false, description: "SnapshotVolumes specifies whether to take snapshots of any PV's referenced in the set of objects included in the Backup."}, {name: "storage_location", kind: String, key: "storageLocation", nilable: true, read_only: false, description: "StorageLocation is a string containing the name of a BackupStorageLocation where the backup should be stored."}, {name: "ttl", kind: String, key: "ttl", nilable: true, read_only: false, description: "TTL is a time.Duration-parseable string describing how long the Backup should be retained for."}, + {name: "uploader_config", kind: ::Hash(String, Int32), key: "uploaderConfig", nilable: true, read_only: false, description: "UploaderConfig specifies the configuration for the uploader."}, {name: "volume_snapshot_locations", kind: ::Array(String), key: "volumeSnapshotLocations", nilable: true, read_only: false, description: "VolumeSnapshotLocations is a list containing names of VolumeSnapshotLocations associated with this backup."}, ] diff --git a/src/crds/velero/v1/backup_status.cr b/src/crds/velero/v1/backup_status.cr index 3d31846..379aa8b 100644 --- a/src/crds/velero/v1/backup_status.cr +++ b/src/crds/velero/v1/backup_status.cr @@ -17,6 +17,7 @@ require "json" {name: "expiration", kind: String, key: "expiration", nilable: true, read_only: false, description: "Expiration is when this Backup is eligible for garbage-collection."}, {name: "failure_reason", kind: String, key: "failureReason", nilable: true, read_only: false, description: "FailureReason is an error that caused the entire backup to fail."}, {name: "format_version", kind: String, key: "formatVersion", nilable: true, read_only: false, description: "FormatVersion is the backup format version, including major, minor, and patch version."}, + {name: "hook_status", kind: ::Hash(String, Int32), key: "hookStatus", nilable: true, read_only: false, description: "HookStatus contains information about the status of the hooks."}, {name: "phase", kind: String, key: "phase", nilable: true, read_only: false, description: "Phase is the current state of the Backup."}, {name: "progress", kind: ::Hash(String, Int32), key: "progress", nilable: true, read_only: false, description: "Progress contains information about the backup's execution progress. Note that this information is best-effort only -- if Velero fails to update it during a backup for any reason, it may be [inaccurate/stale.](inaccurate/stale.)"}, {name: "start_timestamp", kind: String, key: "startTimestamp", nilable: true, read_only: false, description: "StartTimestamp records the time a backup was started. Separate from CreationTimestamp, since that value changes on restores. The server's time is used for StartTimestamps"},