diff --git a/apis/addons/v1alpha1/zz_generated.deepcopy.go b/apis/addons/v1alpha1/zz_generated.deepcopy.go index e1da39968..221335083 100644 --- a/apis/addons/v1alpha1/zz_generated.deepcopy.go +++ b/apis/addons/v1alpha1/zz_generated.deepcopy.go @@ -1,4 +1,5 @@ //go:build !ignore_autogenerated +// +build !ignore_autogenerated /* Copyright AppsCode Inc. and Contributors diff --git a/apis/config/v1alpha1/zz_generated.deepcopy.go b/apis/config/v1alpha1/zz_generated.deepcopy.go index 56bd12184..5dac5c136 100644 --- a/apis/config/v1alpha1/zz_generated.deepcopy.go +++ b/apis/config/v1alpha1/zz_generated.deepcopy.go @@ -1,4 +1,5 @@ //go:build !ignore_autogenerated +// +build !ignore_autogenerated /* Copyright AppsCode Inc. and Contributors diff --git a/apis/constant.go b/apis/constant.go index 43d278dae..33efe3598 100644 --- a/apis/constant.go +++ b/apis/constant.go @@ -75,7 +75,6 @@ const ( KubeStashAppRefKind = "kubestash.com/app-ref-kind" KubeStashAppRefNamespace = "kubestash.com/app-ref-namespace" KubeStashAppRefName = "kubestash.com/app-ref-name" - KubeDBAppVersion = "kubedb.com/db-version" ) // Keys for structure logging @@ -153,3 +152,9 @@ const ( SnapshotVersionV1 = "v1" DirRepository = "repository" ) + +// Annotations +const ( + AnnKubeDBAppVersion = "kubedb.com/db-version" + AnnRestoreSessionBeneficiary = "restoresession.kubestash.com/beneficiary" +) diff --git a/apis/core/v1alpha1/backupconfiguration_types.go b/apis/core/v1alpha1/backupconfiguration_types.go index 105c4ce44..5a34cb25a 100644 --- a/apis/core/v1alpha1/backupconfiguration_types.go +++ b/apis/core/v1alpha1/backupconfiguration_types.go @@ -132,10 +132,10 @@ type SessionConfig struct { // +optional RetryConfig *RetryConfig `json:"retryConfig,omitempty"` - // Timeout specifies the maximum duration of backup. BackupSession will be considered Failed - // if backup does not complete within this time limit. By default, KubeStash don't set any timeout for backup. + // BackupTimeout specifies the maximum duration of backup. Backup will be considered Failed + // if backup tasks do not complete within this time limit. By default, KubeStash don't set any timeout for backup. // +optional - Timeout *metav1.Duration `json:"timeout,omitempty"` + BackupTimeout *metav1.Duration `json:"backupTimeout,omitempty"` // SessionHistoryLimit specifies how many backup Jobs and associate resources KubeStash should keep for debugging purpose. // The default value is 1. diff --git a/apis/core/v1alpha1/backupsession_helpers.go b/apis/core/v1alpha1/backupsession_helpers.go index d89f19e81..edf3152e6 100644 --- a/apis/core/v1alpha1/backupsession_helpers.go +++ b/apis/core/v1alpha1/backupsession_helpers.go @@ -243,3 +243,14 @@ func (b *BackupSession) checkFailureInRetentionPolicy() (bool, string) { } return false, "" } + +func (b *BackupSession) GetRemainingTimeoutDuration() (*metav1.Duration, error) { + if b.Spec.BackupTimeout == nil || b.Status.BackupDeadline == nil { + return nil, nil + } + currentTime := metav1.Now() + if b.Status.BackupDeadline.Before(¤tTime) { + return nil, fmt.Errorf("deadline exceeded") + } + return &metav1.Duration{Duration: b.Status.BackupDeadline.Sub(currentTime.Time)}, nil +} diff --git a/apis/core/v1alpha1/backupsession_types.go b/apis/core/v1alpha1/backupsession_types.go index 5ccf2fafc..9f0948396 100644 --- a/apis/core/v1alpha1/backupsession_types.go +++ b/apis/core/v1alpha1/backupsession_types.go @@ -63,6 +63,11 @@ type BackupSessionSpec struct { // If this set to non-zero, KubeStash will create a new BackupSession if the current one fails. // +optional RetryLeft int32 `json:"retryLeft,omitempty"` + + // BackupTimeout specifies the maximum duration of backup. Backup will be considered Failed + // if backup tasks do not complete within this time limit. By default, KubeStash don't set any timeout for backup. + // +optional + BackupTimeout *metav1.Duration `json:"backupTimeout,omitempty"` } // BackupSessionStatus defines the observed state of BackupSession @@ -75,10 +80,10 @@ type BackupSessionStatus struct { // +optional Duration string `json:"duration,omitempty"` - // Deadline specifies the deadline of backup. BackupSession will be - // considered Failed if backup does not complete within this deadline + // BackupDeadline specifies the deadline of backup. Backup will be + // considered Failed if it does not complete within this deadline // +optional - Deadline *metav1.Time `json:"sessionDeadline,omitempty"` + BackupDeadline *metav1.Time `json:"backupDeadline,omitempty"` // TotalSnapshots specifies the total number of snapshots created for this backupSession. // +optional diff --git a/apis/core/v1alpha1/restoresession_helpers.go b/apis/core/v1alpha1/restoresession_helpers.go index df970cf4f..11121df28 100644 --- a/apis/core/v1alpha1/restoresession_helpers.go +++ b/apis/core/v1alpha1/restoresession_helpers.go @@ -17,6 +17,7 @@ limitations under the License. package v1alpha1 import ( + "fmt" metav1 "k8s.io/apimachinery/pkg/apis/meta/v1" kmapi "kmodules.xyz/client-go/api/v1" "kubestash.dev/apimachinery/apis" @@ -41,8 +42,7 @@ func (rs *RestoreSession) CalculatePhase() RestorePhase { } if cutil.IsConditionTrue(rs.Status.Conditions, TypeMetricsPushed) && - (cutil.IsConditionTrue(rs.Status.Conditions, TypeDeadlineExceeded) || - cutil.IsConditionFalse(rs.Status.Conditions, TypePreRestoreHooksExecutionSucceeded) || + (cutil.IsConditionFalse(rs.Status.Conditions, TypePreRestoreHooksExecutionSucceeded) || cutil.IsConditionFalse(rs.Status.Conditions, TypePostRestoreHooksExecutionSucceeded) || cutil.IsConditionFalse(rs.Status.Conditions, TypeRestoreExecutorEnsured)) { return RestoreFailed @@ -181,3 +181,14 @@ func (rs *RestoreSession) GetDataSourceNamespace() string { } return rs.Spec.DataSource.Namespace } + +func (rs *RestoreSession) GetRemainingTimeoutDuration() (*metav1.Duration, error) { + if rs.Spec.RestoreTimeout == nil || rs.Status.RestoreDeadline == nil { + return nil, nil + } + currentTime := metav1.Now() + if rs.Status.RestoreDeadline.Before(¤tTime) { + return nil, fmt.Errorf("deadline exceeded") + } + return &metav1.Duration{Duration: rs.Status.RestoreDeadline.Sub(currentTime.Time)}, nil +} diff --git a/apis/core/v1alpha1/restoresession_helpers_test.go b/apis/core/v1alpha1/restoresession_helpers_test.go index 2e3b9d8f8..7cf973621 100644 --- a/apis/core/v1alpha1/restoresession_helpers_test.go +++ b/apis/core/v1alpha1/restoresession_helpers_test.go @@ -230,25 +230,6 @@ func TestRestoreSessionPhaseIsFailedIfRestoreExecutorEnsuredConditionIsFalse(t * assert.Equal(t, RestoreFailed, rs.CalculatePhase()) } -func TestRestoreSessionPhaseIsFailedIfDeadlineExceededConditionIsTrue(t *testing.T) { - rs := sampleRestoreSession(func(r *RestoreSession) { - r.Status.Conditions = append(r.Status.Conditions, - kmapi.Condition{ - Type: TypeDeadlineExceeded, - Status: metav1.ConditionTrue, - Reason: ReasonFailedToCompleteWithinDeadline, - }, - kmapi.Condition{ - Type: TypeMetricsPushed, - Status: metav1.ConditionTrue, - Reason: ReasonSuccessfullyPushedMetrics, - }, - ) - }) - - assert.Equal(t, RestoreFailed, rs.CalculatePhase()) -} - func TestRestoreSessionPhaseIsRunningIfPostRestoreHooksNotExecuted(test *testing.T) { rs := sampleRestoreSession(func(r *RestoreSession) { r.Status.Components = map[string]ComponentRestoreStatus{ diff --git a/apis/core/v1alpha1/restoresession_types.go b/apis/core/v1alpha1/restoresession_types.go index f1bfbf52b..5b737d576 100644 --- a/apis/core/v1alpha1/restoresession_types.go +++ b/apis/core/v1alpha1/restoresession_types.go @@ -64,10 +64,10 @@ type RestoreSessionSpec struct { // +optional Hooks *RestoreHooks `json:"hooks,omitempty"` - // Timeout specifies a duration that KubeStash should wait for the session execution to be completed. - // If the session execution does not finish within this time period, KubeStash will consider this session as a failure. + // RestoreTimeout specifies a duration that KubeStash should wait for the restore to be completed. + // If the restore tasks do not finish within this time period, KubeStash will consider this restore as a failure. // +optional - Timeout *metav1.Duration `json:"timeout,omitempty"` + RestoreTimeout *metav1.Duration `json:"restoreTimeout,omitempty"` // ManifestOptions provide options to select particular manifest object to restore // +optional @@ -106,6 +106,10 @@ type ManifestRestoreOptions struct { // ZooKeeper specifies the options for selecting particular ZooKeeper components to restore in manifest restore // +optional ZooKeeper *KubeDBManifestOptions `json:"zooKeeper,omitempty"` + + // Redis specifies the options for selecting particular Redis components to restore in manifest restore + // +optional + Redis *KubeDBManifestOptions `json:"redis,omitempty"` } type MSSQLServerManifestOptions struct { @@ -278,10 +282,10 @@ type RestoreSessionStatus struct { // +optional Duration string `json:"duration,omitempty"` - // Deadline specifies a timestamp till this session is valid. If the session does not complete within this deadline, - // it will be considered as failed. + // RestoreDeadline specifies the deadline of restore. Restore will be + // considered Failed if it does not complete within this deadline // +optional - Deadline *metav1.Time `json:"deadline,omitempty"` + RestoreDeadline *metav1.Time `json:"restoreDeadline,omitempty"` // TotalComponents represents the number of total components for this RestoreSession // +optional diff --git a/apis/core/v1alpha1/types.go b/apis/core/v1alpha1/types.go index 96dbb6521..445f867be 100644 --- a/apis/core/v1alpha1/types.go +++ b/apis/core/v1alpha1/types.go @@ -217,9 +217,6 @@ type RetryConfig struct { } const ( - TypeDeadlineExceeded = "DeadlineExceeded" - ReasonFailedToCompleteWithinDeadline = "FailedToCompleteWithinDeadline" - // TypeMetricsPushed indicates whether Metrics are pushed or not TypeMetricsPushed = "MetricsPushed" ReasonSuccessfullyPushedMetrics = "SuccessfullyPushedMetrics" diff --git a/apis/core/v1alpha1/zz_generated.deepcopy.go b/apis/core/v1alpha1/zz_generated.deepcopy.go index e7d09159e..a1ec3193c 100644 --- a/apis/core/v1alpha1/zz_generated.deepcopy.go +++ b/apis/core/v1alpha1/zz_generated.deepcopy.go @@ -1,4 +1,5 @@ //go:build !ignore_autogenerated +// +build !ignore_autogenerated /* Copyright AppsCode Inc. and Contributors @@ -597,6 +598,11 @@ func (in *BackupSessionSpec) DeepCopyInto(out *BackupSessionSpec) { *out = new(corev1.TypedLocalObjectReference) (*in).DeepCopyInto(*out) } + if in.BackupTimeout != nil { + in, out := &in.BackupTimeout, &out.BackupTimeout + *out = new(metav1.Duration) + **out = **in + } } // DeepCopy is an autogenerated deepcopy function, copying the receiver, creating a new BackupSessionSpec. @@ -612,8 +618,8 @@ func (in *BackupSessionSpec) DeepCopy() *BackupSessionSpec { // DeepCopyInto is an autogenerated deepcopy function, copying the receiver, writing into out. in must be non-nil. func (in *BackupSessionStatus) DeepCopyInto(out *BackupSessionStatus) { *out = *in - if in.Deadline != nil { - in, out := &in.Deadline, &out.Deadline + if in.BackupDeadline != nil { + in, out := &in.BackupDeadline, &out.BackupDeadline *out = (*in).DeepCopy() } if in.TotalSnapshots != nil { @@ -1118,6 +1124,11 @@ func (in *ManifestRestoreOptions) DeepCopyInto(out *ManifestRestoreOptions) { *out = new(KubeDBManifestOptions) (*in).DeepCopyInto(*out) } + if in.Redis != nil { + in, out := &in.Redis, &out.Redis + *out = new(KubeDBManifestOptions) + (*in).DeepCopyInto(*out) + } } // DeepCopy is an autogenerated deepcopy function, copying the receiver, creating a new ManifestRestoreOptions. @@ -1407,8 +1418,8 @@ func (in *RestoreSessionSpec) DeepCopyInto(out *RestoreSessionSpec) { *out = new(RestoreHooks) (*in).DeepCopyInto(*out) } - if in.Timeout != nil { - in, out := &in.Timeout, &out.Timeout + if in.RestoreTimeout != nil { + in, out := &in.RestoreTimeout, &out.RestoreTimeout *out = new(metav1.Duration) **out = **in } @@ -1437,8 +1448,8 @@ func (in *RestoreSessionStatus) DeepCopyInto(out *RestoreSessionStatus) { *out = new(bool) **out = **in } - if in.Deadline != nil { - in, out := &in.Deadline, &out.Deadline + if in.RestoreDeadline != nil { + in, out := &in.RestoreDeadline, &out.RestoreDeadline *out = (*in).DeepCopy() } if in.Components != nil { @@ -1619,8 +1630,8 @@ func (in *SessionConfig) DeepCopyInto(out *SessionConfig) { *out = new(RetryConfig) **out = **in } - if in.Timeout != nil { - in, out := &in.Timeout, &out.Timeout + if in.BackupTimeout != nil { + in, out := &in.BackupTimeout, &out.BackupTimeout *out = new(metav1.Duration) **out = **in } diff --git a/apis/storage/v1alpha1/zz_generated.deepcopy.go b/apis/storage/v1alpha1/zz_generated.deepcopy.go index 2be96888a..c9ea98030 100644 --- a/apis/storage/v1alpha1/zz_generated.deepcopy.go +++ b/apis/storage/v1alpha1/zz_generated.deepcopy.go @@ -1,4 +1,5 @@ //go:build !ignore_autogenerated +// +build !ignore_autogenerated /* Copyright AppsCode Inc. and Contributors diff --git a/apis/zz_generated.deepcopy.go b/apis/zz_generated.deepcopy.go index 40d4e3503..6ee74efd5 100644 --- a/apis/zz_generated.deepcopy.go +++ b/apis/zz_generated.deepcopy.go @@ -1,4 +1,5 @@ //go:build !ignore_autogenerated +// +build !ignore_autogenerated /* Copyright AppsCode Inc. and Contributors diff --git a/crds/core.kubestash.com_backupbatches.yaml b/crds/core.kubestash.com_backupbatches.yaml index 4918fbb46..790559ece 100644 --- a/crds/core.kubestash.com_backupbatches.yaml +++ b/crds/core.kubestash.com_backupbatches.yaml @@ -110,6 +110,12 @@ spec: description: BatchSession specifies the session configuration for the targets. properties: + backupTimeout: + description: BackupTimeout specifies the maximum duration of + backup. Backup will be considered Failed if backup tasks do + not complete within this time limit. By default, KubeStash + don't set any timeout for backup. + type: string hooks: description: Hooks specifies the backup hooks that should be executed before and/or after the backup. @@ -36194,12 +36200,6 @@ spec: type: array type: object type: array - timeout: - description: Timeout specifies the maximum duration of backup. - BackupSession will be considered Failed if backup does not - complete within this time limit. By default, KubeStash don't - set any timeout for backup. - type: string type: object type: array targets: diff --git a/crds/core.kubestash.com_backupblueprints.yaml b/crds/core.kubestash.com_backupblueprints.yaml index 62b405d7c..f745accca 100644 --- a/crds/core.kubestash.com_backupblueprints.yaml +++ b/crds/core.kubestash.com_backupblueprints.yaml @@ -15716,6 +15716,12 @@ spec: type: object type: array type: object + backupTimeout: + description: BackupTimeout specifies the maximum duration + of backup. Backup will be considered Failed if backup + tasks do not complete within this time limit. By default, + KubeStash don't set any timeout for backup. + type: string hooks: description: Hooks specifies the backup hooks that should be executed before and/or after the backup. @@ -37239,12 +37245,6 @@ spec: debugging purpose. The default value is 1. format: int32 type: integer - timeout: - description: Timeout specifies the maximum duration of backup. - BackupSession will be considered Failed if backup does - not complete within this time limit. By default, KubeStash - don't set any timeout for backup. - type: string type: object type: array type: object diff --git a/crds/core.kubestash.com_backupconfigurations.yaml b/crds/core.kubestash.com_backupconfigurations.yaml index 30807c31b..bd2a31650 100644 --- a/crds/core.kubestash.com_backupconfigurations.yaml +++ b/crds/core.kubestash.com_backupconfigurations.yaml @@ -14554,6 +14554,12 @@ spec: type: object type: array type: object + backupTimeout: + description: BackupTimeout specifies the maximum duration of + backup. Backup will be considered Failed if backup tasks do + not complete within this time limit. By default, KubeStash + don't set any timeout for backup. + type: string hooks: description: Hooks specifies the backup hooks that should be executed before and/or after the backup. @@ -34373,12 +34379,6 @@ spec: purpose. The default value is 1. format: int32 type: integer - timeout: - description: Timeout specifies the maximum duration of backup. - BackupSession will be considered Failed if backup does not - complete within this time limit. By default, KubeStash don't - set any timeout for backup. - type: string type: object type: array target: diff --git a/crds/core.kubestash.com_backupsessions.yaml b/crds/core.kubestash.com_backupsessions.yaml index 51f413b2e..fe23c6583 100644 --- a/crds/core.kubestash.com_backupsessions.yaml +++ b/crds/core.kubestash.com_backupsessions.yaml @@ -56,6 +56,12 @@ spec: description: BackupSessionSpec specifies the information related to the respective backup invoker and session. properties: + backupTimeout: + description: BackupTimeout specifies the maximum duration of backup. + Backup will be considered Failed if backup tasks do not complete + within this time limit. By default, KubeStash don't set any timeout + for backup. + type: string invoker: description: Invoker points to the respective BackupConfiguration or BackupBatch which is responsible for triggering this backup. @@ -91,6 +97,11 @@ spec: status: description: BackupSessionStatus defines the observed state of BackupSession properties: + backupDeadline: + description: BackupDeadline specifies the deadline of backup. Backup + will be considered Failed if it does not complete within this deadline + format: date-time + type: string conditions: description: Conditions represents list of conditions regarding this BackupSession @@ -249,12 +260,6 @@ spec: not. This field will exist only if the `retryConfig` has been set in the respective backup invoker. type: boolean - sessionDeadline: - description: Deadline specifies the deadline of backup. BackupSession - will be considered Failed if backup does not complete within this - deadline - format: date-time - type: string snapshots: description: Snapshots specifies the Snapshots status items: diff --git a/crds/core.kubestash.com_restoresessions.yaml b/crds/core.kubestash.com_restoresessions.yaml index 7e9f19187..8f15ba2a9 100644 --- a/crds/core.kubestash.com_restoresessions.yaml +++ b/crds/core.kubestash.com_restoresessions.yaml @@ -24390,6 +24390,60 @@ spec: type: object x-kubernetes-map-type: atomic type: object + redis: + description: Redis specifies the options for selecting particular + Redis components to restore in manifest restore + properties: + authSecret: + description: AuthSecret specifies whether to restore the AuthSecret + manifest or not + type: boolean + authSecretName: + description: AuthSecretName specifies new name of the AuthSecret + yaml after restore + type: string + configSecret: + description: ConfigSecret specifies whether to restore the + ConfigSecret manifest or not + type: boolean + configSecretName: + description: ConfigSecretName specifies new name of the ConfigSecret + yaml after restore + type: string + db: + description: DB specifies whether to restore the DB manifest + or not + type: boolean + dbName: + description: DBName specifies the new name of the DB yaml + after restore + type: string + initScript: + description: InitScript specifies whether to restore the InitScript + manifest or not + type: boolean + tlsIssuerRef: + description: TLSIssuerRef specifies the name of the IssuerRef + used for TLS configurations for both client and server + 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 + type: object restoreNamespace: description: RestoreNamespace specifies the Namespace where the restored files will be applied @@ -24449,6 +24503,12 @@ spec: x-kubernetes-map-type: atomic type: object type: object + restoreTimeout: + description: RestoreTimeout specifies a duration that KubeStash should + wait for the restore to be completed. If the restore tasks do not + finish within this time period, KubeStash will consider this restore + as a failure. + type: string target: description: Target indicates the target application where the data will be restored. The target must be in the same namespace as the @@ -24467,12 +24527,6 @@ spec: required: - name type: object - timeout: - description: Timeout specifies a duration that KubeStash should wait - for the session execution to be completed. If the session execution - does not finish within this time period, KubeStash will consider - this session as a failure. - type: string type: object status: description: RestoreSessionStatus defines the observed state of RestoreSession @@ -24557,12 +24611,6 @@ spec: - type type: object type: array - deadline: - description: Deadline specifies a timestamp till this session is valid. - If the session does not complete within this deadline, it will be - considered as failed. - format: date-time - type: string dependencies: description: Dependencies specifies whether the objects required by this RestoreSession exist or not @@ -24667,6 +24715,11 @@ spec: - Invalid - Unknown type: string + restoreDeadline: + description: RestoreDeadline specifies the deadline of restore. Restore + will be considered Failed if it does not complete within this deadline + format: date-time + type: string targetFound: description: TargetFound specifies whether the restore target exist or not diff --git a/pkg/restic/commands.go b/pkg/restic/commands.go index a9c1a0a2f..9d37aeb80 100644 --- a/pkg/restic/commands.go +++ b/pkg/restic/commands.go @@ -35,7 +35,8 @@ import ( ) const ( - ResticCMD = "restic" + ResticCMD = "restic" + TimeoutCMD = "timeout" ) type Snapshot struct { @@ -176,7 +177,10 @@ func (w *ResticWrapper) backup(params backupParams) ([]byte, error) { args = w.appendInsecureTLSFlag(args) args = w.appendMaxConnectionsFlag(args) - return w.run(Command{Name: ResticCMD, Args: args}) + command := Command{Name: ResticCMD, Args: args} + command = w.wrapWithTimeoutIfConfigured(command) + + return w.run(command) } func (w *ResticWrapper) backupFromStdin(options BackupOptions) ([]byte, error) { @@ -200,7 +204,10 @@ func (w *ResticWrapper) backupFromStdin(options BackupOptions) ([]byte, error) { args = w.appendInsecureTLSFlag(args) args = w.appendMaxConnectionsFlag(args) - commands = append(commands, Command{Name: ResticCMD, Args: args}) + command := Command{Name: ResticCMD, Args: args} + command = w.wrapWithTimeoutIfConfigured(command) + + commands = append(commands, command) return w.run(commands...) } @@ -246,7 +253,10 @@ func (w *ResticWrapper) restore(params restoreParams) ([]byte, error) { args = w.appendInsecureTLSFlag(args) args = w.appendMaxConnectionsFlag(args) - return w.run(Command{Name: ResticCMD, Args: args}) + command := Command{Name: ResticCMD, Args: args} + command = w.wrapWithTimeoutIfConfigured(command) + + return w.run(command) } func (w *ResticWrapper) DumpOnce(dumpOptions DumpOptions) ([]byte, error) { @@ -277,10 +287,11 @@ func (w *ResticWrapper) DumpOnce(dumpOptions DumpOptions) ([]byte, error) { args = w.appendMaxConnectionsFlag(args) args = w.appendInsecureTLSFlag(args) + command := Command{Name: ResticCMD, Args: args} + command = w.wrapWithTimeoutIfConfigured(command) + // first add restic command, then add StdoutPipeCommands - commands := []Command{ - {Name: ResticCMD, Args: args}, - } + commands := []Command{command} commands = append(commands, dumpOptions.StdoutPipeCommands...) return w.run(commands...) } @@ -368,7 +379,7 @@ func (w *ResticWrapper) run(commands ...Command) ([]byte, error) { w.sh.Stderr = io.MultiWriter(os.Stderr, errBuff) for _, cmd := range commands { - if cmd.Name == ResticCMD { + if cmd.Name == ResticCMD || cmd.Name == TimeoutCMD { // first apply NiceSettings, then apply IONiceSettings cmd, err = w.applyNiceSettings(cmd) if err != nil { @@ -393,6 +404,9 @@ func (w *ResticWrapper) run(commands ...Command) ([]byte, error) { func formatError(err error, stdErr string) error { parts := strings.Split(strings.TrimSuffix(stdErr, "\n"), "\n") if len(parts) > 1 { + if strings.Contains(parts[1], "signal terminated") { + return errors.New(strings.Join(append([]string{"deadline exceeded or signal terminated"}, parts[2:]...), " ")) + } return errors.New(strings.Join(parts[1:], " ")) } return err @@ -515,3 +529,12 @@ func (w *ResticWrapper) removeKey(params keyParams) ([]byte, error) { return w.run(Command{Name: ResticCMD, Args: args}) } + +func (w *ResticWrapper) wrapWithTimeoutIfConfigured(cmd Command) Command { + if w.config.Timeout != nil { + timeoutArgs := []interface{}{fmt.Sprintf("%f", w.config.Timeout.Seconds()), cmd.Name} + timeoutArgs = append(timeoutArgs, cmd.Args...) + return Command{Name: TimeoutCMD, Args: timeoutArgs} + } + return cmd +} diff --git a/pkg/restic/config.go b/pkg/restic/config.go index 6bf9617b6..2a910c3b5 100644 --- a/pkg/restic/config.go +++ b/pkg/restic/config.go @@ -18,12 +18,12 @@ package restic import ( "fmt" + metav1 "k8s.io/apimachinery/pkg/apis/meta/v1" + "kubestash.dev/apimachinery/apis/storage/v1alpha1" "os" "path/filepath" "sort" - "kubestash.dev/apimachinery/apis/storage/v1alpha1" - shell "gomodules.xyz/go-sh" core "k8s.io/api/core/v1" kmapi "kmodules.xyz/client-go/api/v1" @@ -103,6 +103,7 @@ type SetupOptions struct { EnableCache bool Nice *ofst.NiceSettings IONice *ofst.IONiceSettings + Timeout *metav1.Duration } type KeyOptions struct { diff --git a/pkg/restic/restic_test.go b/pkg/restic/restic_test.go index b9dd6b97b..c4b45ae37 100644 --- a/pkg/restic/restic_test.go +++ b/pkg/restic/restic_test.go @@ -18,14 +18,6 @@ package restic import ( "fmt" - "os" - "path/filepath" - "testing" - - addonapi "kubestash.dev/apimachinery/apis/addons/v1alpha1" - coreapi "kubestash.dev/apimachinery/apis/core/v1alpha1" - storageapi "kubestash.dev/apimachinery/apis/storage/v1alpha1" - "github.com/stretchr/testify/assert" "gomodules.xyz/pointer" core "k8s.io/api/core/v1" @@ -35,8 +27,15 @@ import ( kmapi "kmodules.xyz/client-go/api/v1" storage "kmodules.xyz/objectstore-api/api/v1" ofst "kmodules.xyz/offshoot-api/api/v1" + addonapi "kubestash.dev/apimachinery/apis/addons/v1alpha1" + coreapi "kubestash.dev/apimachinery/apis/core/v1alpha1" + storageapi "kubestash.dev/apimachinery/apis/storage/v1alpha1" + "os" + "path/filepath" "sigs.k8s.io/controller-runtime/pkg/client" "sigs.k8s.io/controller-runtime/pkg/client/fake" + "testing" + "time" ) var ( @@ -464,6 +463,38 @@ func TestBackupRestoreWithArgs(t *testing.T) { } } +func TestBackupWithTimeout(t *testing.T) { + tempDir, err := os.MkdirTemp("", "stash-unit-test-") + if err != nil { + t.Error(err) + return + } + + w, err := setupTest(tempDir) + if err != nil { + t.Error(err) + return + } + defer cleanup(tempDir) + + // Initialize Repository + err = w.InitializeRepository() + if err != nil { + t.Error(err) + return + } + + duration := metav1.Duration{Duration: 10 * time.Millisecond} + w.config.Timeout = &duration + + backupOpt := BackupOptions{ + StdinPipeCommands: []Command{stdinPipeCommand}, + StdinFileName: fileName, + } + _, err = w.RunBackup(backupOpt) + assert.Error(t, err, "Timeout error") +} + func TestVerifyRepositoryIntegrity(t *testing.T) { tempDir, err := os.MkdirTemp("", "stash-unit-test-") if err != nil {